五分钟理解keep alive用法及原理
引言
keep-alive
可以实现组件缓存,当组件切换时不会对当前组件进行卸载。
主要是有include、exclude、max
三个属性;
前两个属性允许keep-alive
有条件的进行缓存;
max
可以定义组件最大的缓存个数,如果超过了这个个数的话,在下一个新实例创建之前,就会将以缓存组件中最久没有被访问到的实例销毁掉。
两个生命周期activated/deactivated
,用来得知当前组件是否处于活跃状态。
Home.vue
<template> <div class="hello"> <h1>你怎么还在学习?🗑️</h1> <input placeholder="输入框" /> <router-link to="/about">我的人生理想</router-link> </div> </template>
About.vue
<template> <div class="hello"> <h1>我想取老婆 🐜</h1> </div> </template>
App.vue
<template> <div id="app"> <router-view/> </div> </template> <script> import Home from './views/Home.vue' import About from './views/About.vue' export default { components: { Home, About, } } </script>
你在输入框中输入信息,点击跳转到另外一个页面后,回到该页面,你会发现,输入框中的文字消失了。怎么办勒
使用 keep-alive 包裹 router-view,同时指定需要缓存的组件名称。(PS:请在要缓存的组件中,写明 name 属性,并赋值。不然缓存不生效)
Home.vue
App.vue
<template> <div id="app"> <keep-alive include="Home"> <router-view/> </keep-alive> </div> </template> <script> import Home from './views/Home.vue' import About from './views/About.vue' export default { components: { Home, About, } } </script>
结合 Router,缓存页面
router.js
import Vue from 'vue' import Router from 'vue-router' import Home from './views/Home.vue' Vue.use(Router) export default new Router({ routes: [ { path: '/', name: 'home', component: Home, // 需要被缓存 meta: { keepAlive: true } }, { path: '/about', name: 'about', component: () => import('./views/About.vue') }, ] })
Home.vue
<script> export default { name: "Home", beforeRouteLeave(to, from, next) { to.meta.keepAlive = true; next(); }, }; </script>
App.vue
<template> <div id="app"> <keep-alive> <router-view v-if="$route.meta.keepAlive"></router-view> </keep-alive> <router-view v-if="!$route.meta.keepAlive"></router-view> </div> </template>
剩下的一些其他特性,可以自行前往官网查阅 cn.vuejs.org/v2/api/#kee…
keep-alive 原理
keep-alive中运用了LRU(Least Recently Used)
算法。
- 获取
keep-alive
包裹着的第一个子组件对象及其组件名; 如果 keep-alive 存在多个子元素,keep-alive
要求同时只有一个子元素被渲染。所以在开头会获取插槽内的子元素,调用getFirstComponentChild
获取到第一个子元素的VNode
。 - 根据设定的黑白名单(如果有)进行条件匹配,决定是否缓存。不匹配,直接返回组件实例(
VNode
),否则开启缓存策略。 - 根据组件ID和tag生成缓存Key,并在缓存对象中查找是否已缓存过该组件实例。如果存在,直接取出缓存值并更新该key在
this.keys
中的位置(更新key的位置是实现LRU
置换策略的关键)。 - 如果不存在,则在
this.cache
对象中存储该组件实例并保存key值,之后检查缓存的实例数量是否超过max设置值,超过则根据LRU
置换策略删除最近最久未使用的实例(即是下标为0的那个key)。最后将该组件实例的keepAlive
属性值设置为true
。
var KeepAlive = { name: 'keep-alive', // 抽象组件 abstract: true, // 接收的参数 props: { include: patternTypes, exclude: patternTypes, max: [String, Number] }, // 创建缓存表 created: function created () { this.cache = Object.create(null); this.keys = []; }, destroyed: function destroyed () { for (var key in this.cache) { pruneCacheEntry(this.cache, key, this.keys); } }, mounted: function mounted () { var this$1 = this; this.$watch('include', function (val) { pruneCache(this$1, function (name) { return matches(val, name); }); }); this.$watch('exclude', function (val) { pruneCache(this$1, function (name) { return !matches(val, name); }); }); }, render: function render () { var slot = this.$slots.default; // 获取 `keep-alive` 包裹着的第一个子组件对象及其组件名; // 如果 keep-alive 存在多个子元素,`keep-alive` 要求同时只有一个子元素被渲染。 // 所以在开头会获取插槽内的子元素, // 调用 `getFirstComponentChild` 获取到第一个子元素的 `VNode`。 var vnode = getFirstComponentChild(slot); var componentOptions = vnode && vnode.componentOptions; if (componentOptions) { // check pattern var name = getComponentName(componentOptions); var ref = this; var include = ref.include; var exclude = ref.exclude; // 根据设定的黑白名单(如果有)进行条件匹配,决定是否缓存。 if ( // not included (include && (!name || !matches(include, name))) || // excluded (exclude && name && matches(exclude, name)) ) { // 不匹配,直接返回组件实例(`VNode`),否则开启缓存策略。 return vnode } var ref$1 = this; var cache = ref$1.cache; var keys = ref$1.keys; // 根据组件ID和tag生成缓存Key var key = vnode.key == null ? componentOptions.Ctor.cid + (componentOptions.tag ? ("::" + (componentOptions.tag)) : '') : vnode.key; if (cache[key]) { // 并在缓存对象中查找是否已缓存过该组件实例。如果存在,直接取出缓存值 vnode.componentInstance = cache[key].componentInstance; // 并更新该key在this.keys中的位置(更新key的位置是实现LRU置换策略的关键)。 remove(keys, key); keys.push(key); } else { // 如果不存在,则在this.cache对象中存储该组件实例并保存key值, cache[key] = vnode; keys.push(key); // 之后检查缓存的实例数量是否超过max设置值,超过则根据LRU置换策略删除最近最久未使用的实例 if (this.max && keys.length > parseInt(this.max)) { pruneCacheEntry(cache, keys[0], keys, this._vnode); } } // 最后将该组件实例的keepAlive属性值设置为true。 vnode.data.keepAlive = true; } return vnode || (slot && slot[0]) } };
以上就是五分钟理解keep alive用法及原理的详细内容,更多关于keep alive用法原理的资料请关注脚本之家其它相关文章!
相关文章
使用vue-element-admin框架从后端动态获取菜单功能的实现
​ vue-element-admin是一个纯前端的框架,左侧菜单是根据路由生成的。实际开发中经常需要根据当前登陆人员的信息从后端获取菜单进行展示,本文将详细介绍如何实现该功能2021-04-04在vue-cli 3中给stylus、sass样式传入共享的全局变量
这篇文章主要介绍了在vue-cli 3中, 给stylus、sass样式传入共享的全局变量,本文给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下2019-08-08keep-alive include和exclude无效问题及解决
这篇文章主要介绍了keep-alive include和exclude无效问题及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教2023-11-11
最新评论