beforeRouteEnter 异步获取数据给实例问题

场景:vue-router路由钩子 beforeRouteEnter可以用来在初始进入页面前,http 异步获取数据mockData,预先判断是进入 A 页、还是 B 页,还是留在本页;而如果留在本页的话,还需要在mounted根据mockData来判断显示哪种状态(可以在本页面实例创建后,重新发起 http 请求获取mockData,但是没有必要,造成代码冗余);

执行顺序:

 async beforeRouteEnter(to, from, next) {
      let res = await gameData()
      console.log('beforeRouteEnter start');
      next(vm => {
        console.log("vm start")
        vm.is_exchange = res.is_exchange
        vm.is_finish = res.is_finish
      })
 },
 beforeCreate() {
    console.log("beforeCreate start")
 },
 mounted(){
   console.log('mounted start');
   if(this.is_finish){
     this.modalMsg="活动已结束"
     return;
   }
   if(this.is_exchange){
     this.modalMsg="您已兑换奖品"
     return;
   }
 }

打印结果如下:

beforeRouteEnter start

beforeCreate start

mounted start

vm start

由打印结果,我们可以总结 beforeRouteEnter 钩子确实在 vue 实例创建前执行,但是其 next 函数中 vm 回调不是同步执行,而是等到 mounted 执行完之后,才执行

症结: 因为我们要根据mockData中的is_exchangeis_finish参数来判断决定页面初始状态,此过程需要在mounted中执行;但是 mounted 执行时,vm 还未执行,即 mounted 拿不到
is_exchangeis_finish 这两个值,这样就造成了冲突;

解决:next 中 打印 vm,发现 vm 就是当前 vue 实例对象,即可以使用 vm 调用所有当前实例的变量和方法;那依次,能否将判断逻辑写入 methods 中一个方法中,使用 vm来调用呢?

async beforeRouteEnter(to, from, next) {
      let res = await gameData()
      console.log('beforeRouteEnter start');
      next(vm => {
        console.log("vm start")
        vm.is_exchange = res.is_exchange
        vm.is_finish = res.is_finish

        vm.judge();//赋值之后,此处调用判断方法
      })
 },
 beforeCreate() {
    console.log("beforeCreate start")
 },
 mounted(){
   console.log('mounted start');
 },
 methods:{
   judge(){// 将判断逻辑写入judge方法
    if(this.is_finish){
      this.modalMsg="活动已结束"
      return;
    }
    if(this.is_exchange){
      this.modalMsg="您已兑换奖品"
      return;
    }
   }
 }

至此,问题解决。有同学可能会问,在 vm 中调用 judge 方法时,mounted 已执行,此时页面已渲染,再去判断初始状态,会不会有闪烁问题?本人经过测试,发现不会,据此推断,在 mounted 执行结束之后,页面没有开始更新动作,而是在执行完 vm 回调之后,再去渲染。这样的话,逻辑上就行得通了,但是这个只是推断,还需要在源码层面找到依据才可以。

github地址

要做前端架构师的正在前行的人

1.7k 声望
63 粉丝
0 条评论
推荐阅读
日常总结2
使用proxy,代理一个空对象{}时,如果在get方法中,返回的属性是{}没有的,底层就会调用getOwnPropertyDescriptor属性,进而出现问题,比如报如下问题

前端熟练工阅读 580

Vue中的diff算法
diff算法是一种通过同层的树节点进行比较的高效算法,避免了对树进行逐层搜索遍历,所以时间复杂度只有 O(n)。diff算法的在很多场景下都有应用,例如在 vue 虚拟 dom 渲染成真实 dom 的新旧 VNode 节点比较更新时...

款冬27阅读 13.4k评论 7

给我实现一个前端的 Excel 导入和导出功能
前言【负责人 A】:现在报表部分基于接口的 Excel 的导入和导出功能有点慢,前端这边能不能实现一下这个功能,然后我们在比对看看效果!【切图仔 B】: 接口这边不能优化一下吗?比如排查下慢的原因什么的。【负...

熊的猫19阅读 2.5k

封面图
vue-property-decorator使用手册
@Component 装饰器可以接收一个对象作为参数,可以在对象中声明 components ,filters,directives等未提供装饰器的选项,也可以声明computed,watch等

似曾相识17阅读 29.1k评论 7

一个开源vue网站博客,nuxt开源网站,前后端分离项目
开媛笔记,基于nuxt ssr首屏服务器端渲染 。用于分享、记录、交流和学习,希望可以帮助到小伙伴们。同时网站在不断更新,创造属于猿(媛)的世界 -$Bao Yalong ..Let's Go! [链接]

jigsaw16阅读 8.4k评论 3

你知道前端水印功能是怎么实现的吗?
前一段时间由于项目需要实现水印功能,于是去了解了相关的内容后,基于 Vue 的实现了一个 v-watermark 指令完成了对应的功能,其实整体内容并不复杂!

熊的猫14阅读 1.7k

封面图
2022 你还不会微前端吗 (上) — 从巨石应用到微应用
微前端系列分为 上/下 两篇,本文为 上篇 主要还是了解微前端的由来、概念、作用等,以及基于已有的微前端框架进行实践,并了解微前端的核心功能所在,而在下篇 2022 你还不会微前端吗 (下) — 揭秘微前端核心原理...

熊的猫14阅读 1.6k

封面图

要做前端架构师的正在前行的人

1.7k 声望
63 粉丝
宣传栏