在平常开发中,有部分组件没有必要多次初始化,这时,我们需要将组件进行持久化,使组件的状态维持不变,在下一次展示时,也不会进行重新初始化组件。
Keep-Alive是Vue.js中的一个内置组件,它可以用来缓存不活动的组件实例,而不是销毁它们。当Keep-Alive包裹动态组件时,它会将组件的状态保存在内存中,以防止在组件切换过程中重复渲染DOM,从而提高性能和用户体验。Keep-Alive自身不会渲染一个DOM元素,也不会出现在父组件链中。
在Keep-Alive组件中,你可以使用props来控制哪些组件需要被缓存,以及可以缓存多少个组件实例。例如,你可以使用include和exclude props来指定需要缓存的组件名称或正则表达式,以及使用max props来限制最多可以缓存的组件实例数量。
当组件在Keep-Alive内被切换时,它的activated和deactivated生命周期钩子函数将会被对应执行。这意味着当组件从缓存中激活时,activated函数会被调用,而当组件被停用时,deactivated函数会被调用。
因此,在使用Keep-Alive时需要注意权衡其优缺点,根据实际需求和情况做出决策。
keepalive 可以接收3个属性做为参数进行匹配对应的组件进行缓存:
它会根据组件的 name 选项进行匹配,所以组件如果想要条件性地被 KeepAlive 缓存,就必须显式声明一个 name 选项。
被keepalive包含的组件不会被再次初始化,也就意味着不会重走生命周期函数
但是有时候是希望我们缓存的组件可以能够再次进行渲染,这时 Vue 为我们解决了这个问题
被包含在 keep-alive 中创建的组件,会多出两个生命周期的钩子: activated 与 deactivated:
activated 在组件挂载时也会调用,并且 deactivated 在组件卸载时也会调用。
这两个钩子不仅适用于 缓存的根组件,也适用于缓存树中的后代组件。
页面第一次进入,钩子的触发顺序created-> mounted-> activated,
退出时触发 deactivated
当再次进入(前进或者后退)时,只触发 activated事件挂载的方法等,
只执行一次的放在 mounted 中;
组件每次进去执行的方法放在 activated 中;
export default {
mounted(){
//只执行一次
},
activated() {
// 每次进去都执行
},
deactivated() {
// 组件卸载时调用
}
}
遇见 vue-router 结合router使用,缓存部分页面
1、用 include (exclude例子类似)
缺点:需要知道组件的 name,项目复杂的时候不是很好的选择
2、使用 meta 属性
优点:不需要例举出需要被缓存组件名称
使用$route.meta的keepAlive属性:
需要在router中设置router的元信息meta:
//...router.js
export default new Router({
routes: [
{
path: '/Hello,',
name: 'Hello',
component: Hello,
meta: {
keepAlive: false // 不需要缓存
}
},
{
path: '/page1',
name: 'Page1',
component: Page1,
meta: {
keepAlive: true // 需要被缓存
}
}
]
})
需要注意的是,include和exclude属性中的组件名字必须是组件定义的name属性值,而不是路由里面定义的名字。
假设这里有 3 个路由: A、B、C。
默认显示 A
B 跳到 A,A 不刷新
C 跳到 A,A 刷新
1、在 A 路由里面设置 meta 属性:
{
path: '/',
name: 'A',
component: A,
meta: {
keepAlive: true // 需要被缓存
}
}
2、在 B 组件里面设置 beforeRouteLeave:
export default {
data() {
return {};
},
methods: {},
beforeRouteLeave(to, from, next) {
// 设置下一个路由的 meta
to.meta.keepAlive = true; // 让 A 缓存,即不刷新
next();
}
};
3、在 C 组件里面设置 beforeRouteLeave:
export default {
data() {
return {};
},
methods: {},
beforeRouteLeave(to, from, next) {
// 设置下一个路由的 meta
to.meta.keepAlive = false; // 让 A 不缓存,即刷新
next();
}
};
这样便能实现 B 回到 A,A 不刷新;而 C 回到 A 则刷新。