详解vue beforeRouteEnter 异步获取数据给实例问题
场景: vue-router路由钩子 beforeRouteEnter可以用来在初始进入页面前,http异步获取数据mockData,预先判断是进入A页、还是B页,还是留在本页;而如果留在本页的话,还需要在mounted根据 mockData来判断显示哪种状态(可以在本页面实例创建后,重新发起http请求获取 mockData,但是没有必要,造成代码冗余);
执行顺序:
asyncbeforeRouteEnter(to,from,next){
letres=awaitgameData()
console.log('beforeRouteEnterstart');
next(vm=>{
console.log("vmstart")
vm.is_exchange=res.is_exchange
vm.is_finish=res.is_finish
})
},
beforeCreate(){
console.log("beforeCreatestart")
},
mounted(){
console.log('mountedstart');
if(this.is_finish){
this.modalMsg="活动已结束"
return;
}
if(this.is_exchange){
this.modalMsg="您已兑换奖品"
return;
}
}
打印结果如下:
beforeRouteEnterstart
beforeCreatestart
mountedstart
vmstart
由打印结果,我们可以总结 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来调用呢?
asyncbeforeRouteEnter(to,from,next){
letres=awaitgameData()
console.log('beforeRouteEnterstart');
next(vm=>{
console.log("vmstart")
vm.is_exchange=res.is_exchange
vm.is_finish=res.is_finish
vm.judge();//赋值之后,此处调用判断方法
})
},
beforeCreate(){
console.log("beforeCreatestart")
},
mounted(){
console.log('mountedstart');
},
methods:{
judge(){//将判断逻辑写入judge方法
if(this.is_finish){
this.modalMsg="活动已结束"
return;
}
if(this.is_exchange){
this.modalMsg="您已兑换奖品"
return;
}
}
}
至此,问题解决。有同学可能会问,在 vm中调用 judge方法时, mounted已执行,此时页面已渲染,再去判断初始状态,会不会有闪烁问题?本人经过测试,发现不会,据此推断,在 mounted执行结束之后,页面没有开始更新动作,而是在执行完 vm回调之后,再去渲染。这样的话,逻辑上就行得通了,但是这个只是推断,还需要在源码层面找到依据才可以。
github地址
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。