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

时间:2021-05-25

场景: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_exchange 和is_finish 参数来判断决定页面初始状态,此过程需要在mounted 中执行;但是mounted 执行时,vm 还未执行,即mounted 拿不到

is_exchange 和is_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地址

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。

相关文章