1.vue官网有关生命周期的图片
2.详细列表
在vue实例的整个生命周期的各个阶段,会提供不同的钩子函数以供我们进行不同的操作。先列出vue官网上对各个钩子函数的详细解析。
生命周期钩子? ?? | 详细 |
---|
beforeCreate | 在实例初始化之后,数据观测(data observer) 和 event/watcher 事件配置之前被调用。 | created | 实例已经创建完成之后被调用。在这一步,实例已完成以下的配置:数据观测(data observer),属性和方法的运算, watch/event 事件回调。然而,挂载阶段还没开始,$el 属性目前不可见。 | beforeMount | 在挂载开始之前被调用:相关的 render 函数首次被调用。 | mounted | el 被新创建的 vm.el 替换,并挂载到实例上去之后调用该钩子。如果 root 实例挂载了一个文档内元素,当 mounted 被调用时 vm.el替换,并挂载到实例上去之后调用该钩子。如果root实例挂载了一个文档内元素,当mounted被调用时vm.el 也在文档内。 | beforeUpdate | 数据更新时调用,发生在虚拟 DOM 重新渲染和打补丁之前。你可以在这个钩子中进一步地更改状态,这不会触发附加的重渲染过程。 | updated | 由于数据更改导致的虚拟 DOM 重新渲染和打补丁,在这之后会调用该钩子。当这个钩子被调用时,组件 DOM 已经更新,所以你现在可以执行依赖于 DOM 的操作。 | activated | keep-alive 组件激活时调用。 | deactivated | keep-alive 组件停用时调用。 | beforeDestroy | 实例销毁之前调用。在这一步,实例仍然完全可用。 | destroyed | Vue 实例销毁后调用。调用后,Vue 实例指示的所有东西都会解绑定,所有的事件监听器会被移除,所有的子实例也会被销毁。 |
3.示例代码钩子执行:
// 子组件Child1
<template>
<div>给我一个div</div>
</template>
<script>
export default {
beforeCreate() {
console.log('beforeCreate 执行了')
},
created() {
console.log('created 执行了')
},
beforeMount() {
console.log('beforeMount 执行了')
},
mounted() {
console.log('mounted 执行了')
},
beforeUpdate() {
console.log('beforeUpdate 执行了')
},
updated() {
console.log('updated 执行了')
},
beforeDestroy() {
console.log('beforeDestroy 执行了')
},
destroyed() {
console.log('destroyed 执行了')
},
}
</script>
// 父组件App
<template>
<div>
<button @click="flag = !flag">点击销毁创建子组件</button>
<Child1 v-if="flag" />
</div>
</template>
<script>
import Child1 from './components//Child1'
export default {
data() {
return {
flag: true,
}
},
components: {
Child1,
},
}
</script>
复制代码
1.启动项目第一次加载页面执行的钩子:
2.点击按钮销毁Child1组件执行的钩子:
Vue - 路由守卫(路由的生命周期)
1.路由导航守卫定义
官方解释:
“导航”表示路由正在发生改变。正如其名,vue-router提供的导航守卫主要用来通过跳转或取消的方式守卫导航。有多种机会植入路由导航过程中:全局的, 单个路由独享的, 或者组件级的。
简单的说,导航守卫就是路由跳转过程中的一些钩子函数。路由跳转是一个大的过程,这个大的过程分为跳转前中后等等细小的过程,在每一个过程中都有一函数,这个函数能让你操作一些其他的事儿,这就是导航守卫。类似于组件生命周期钩子函数
2.路由守卫分类
【1】全局守卫:?是指路由实例上直接操作的钩子函数,特点是所有路由配置的组件都会触发,直白点就是触发路由就会触发这些钩子函数
- beforeEach(to,from, next)
- beforeResolve(to,from, next)
- afterEach(to,from)
【2】路由守卫:?是指在单个路由配置的时候也可以设置的钩子函数
- beforeEnter(to,from, next)
【3】组件守卫:?是指在组件内执行的钩子函数,类似于组件内的生命周期,相当于为配置路由的组件添加的生命周期钩子函数。
- beforeRouteEnter(to,from, next)
- beforeRouteUpdate(to,from, next)
- beforeRouteLeave(to,from, next)
3.路由守卫回调参数介绍
to:即将要进入的目标路由对象;
from:即将要离开的路由对象;
next:涉及到next参数的钩子函数,必须调用next()方法来resolve这个钩子,否则路由会中断在这,不会继续往下执行
- next():进行管道中的下一个钩子。如果全部钩子执行完了,则导航的状态就是confirmed(确认的)。
- next( false )中断当前的导航。如果浏览器的 URL 改变了 (可能是用户手动或者浏览器后退按钮),那么 URL 地址会重置到from路由对应的地址。
- next( ' / ')或者next({ paht:' / ' }):跳转到一个不同的地址。当前的导航被中断,然后进行一个新的导航。可传递的参数可以是router-link标签中的to属性参数或router.push中的选项
- next( error ):如果传入next的参数是一个Error实例,则导航会被终止且该错误会被传递给router.onError()注册过的回调。
4.路由守卫详解
【1】全局前置守卫(beforeEach):?在路由跳转前触发,这个钩子作用主要是用于登录验证,也就是路由还没跳转提前告知,以免跳转了再通知就为时已晚。
const router = new VueRouter({ ... })
?
router.beforeEach((to, from, next) => {
// ...
})
复制代码
【2】全局解析守卫(beforeResolve):?这个钩子和beforeEach类似,也是路由跳转前触发,区别是在导航被确认之前,同时在所有组件内守卫和异步路由组件被解析之后,即在 beforeEach 和 组件内beforeRouteEnter 之后,afterEach之前调用。
【3】全局后置钩子(afterEach):?和beforeEach相反,它是在路由跳转完成后触发,它发生在beforeEach和beforeResolve之后,beforeRouteEnter(组件内守卫)之前。这些钩子不会接受next函数也不会改变导航本身
router.afterEach((to, from) => {
// ...
})
复制代码
【4】路由独享守卫(beforeEnter):?和beforeEach完全相同,如果两个都设置了,beforeEnter则在beforeEach之后紧随执行。在路由配置上直接定义beforeEnter守卫
const router = new VueRouter({
routes: [
? {
? ? path: '/foo',
? ? component: Foo,
? ? beforeEnter: (to, from, next) => {
? ? ? // ...
? ? }
? }
]
})
复制代码
【5】组件内的守卫:
<template>
...
</template>
<script>
export default{
data(){
? //...
},
beforeRouteEnter (to, from, next) {
? // 在渲染该组件的对应路由被 confirm 前调用
? // 不!能!获取组件实例 `this`
? // 因为当守卫执行前,组件实例还没被创建
},
beforeRouteUpdate (to, from, next) {
? // 在当前路由改变,但是该组件被复用时调用
? // 举例来说,对于一个带有动态参数的路径 /foo/:id,在 /foo/1 和 /foo/2 之间跳转的时候,
? // 由于会渲染同样的 Foo 组件,因此组件实例会被复用。而这个钩子就会在这个情况下被调用。
? // 可以访问组件实例 `this`
},
beforeRouteLeave (to, from, next) {
? // 导航离开该组件的对应路由时调用
? // 可以访问组件实例 `this`
}
}
</script>
<style>
...
</style>
复制代码
1. beforeRouteEnter:?该钩子在全局守卫beforeEach和独享守卫beforeEnter之后,全局beforeResolve和全局afterEach之前调用,要注意的是该守卫内访问不到组件的实例,也就是this为undefined。因为它在组件生命周期beforeCreate阶段触发,此时的新组件还没有被创建。在这个钩子函数中,可以通过传一个回调给 next来访问组件实例。在导航被确认的时候执行回调,并且把组件实例作为回调方法的参数。
beforeRouteEnter (to, from, next) {
next(vm => {
? // 通过 `vm` 访问组件实例
})
}
复制代码
2. beforeRouteUpdate:?在当前路由改变时,并且该组件被复用时调用,可以通过this访问实例。
3. beforeRouteLeave:?导航离开该组件的对应路由时调用,可以访问组件实例this。这个离开守卫通常用来禁止用户在还未保存修改前突然离开。该导航可以通过next( false )来取消。
beforeRouteLeave (to, from , next) {
const answer = window.confirm('Do you really want to leave? you have unsaved changes!')
if (answer) {
? next()
} else {
? next(false)
}
}
复制代码
5.完整的导航解析流程
- 触发进入其它路由
- 调用要离开路由的组件守卫beforeRouteLeave
- 调用全局的前置守卫beforeEach
- 在重用的组件里调用?beforeRouteUpdate
- 在路由配置里调用?beforeEnter
- 解析异步路由组件
- 在将要进入的路由组件中调用beforeRouteEnter
- 调用全局的解析守卫beforeResolve
- 导航被确认
- 调用全局的后置钩子afterEach。
- 触发 DOM 更新mounted。
- 执行beforeRouteEnter守卫中传给?next的回调函数。
|