深入理解Vue keep-alive及实践总结


Posted in Javascript onAugust 21, 2019

什么是 keep-alive

在平常开发中,有部分组件没有必要多次初始化,这时,我们需要将组件进行持久化,使组件的状态维持不变,在下一次展示时,也不会进行重新初始化组件。

也就是说,keepalive 是 Vue 内置的一个组件,可以使被包含的组件保留状态,或避免重新渲染 。也就是所谓的组件缓存
<keep-alive>是Vue的内置组件,能在组件切换过程中将状态保留在内存中,防止重复渲染DOM。

<keep-alive> 包裹动态组件时,会缓存不活动的组件实例,而不是销毁它们。和 <transition> 相似,<keep-alive> 是一个抽象组件:它自身不会渲染一个 DOM 元素,也不会出现在父组件链中。

prop:

  • include: 字符串或正则表达式。只有匹配的组件会被缓存。
  • exclude: 字符串或正则表达式。任何匹配的组件都不会被缓存。

keep-alive的声明周期执行

页面第一次进入,钩子的触发顺序

created-> mounted-> activated,退出时触发 deactivated

当再次进入(前进或者后退)时,只触发 activated

事件挂载的方法等,只执行一次的放在 mounted 中;组件每次进去执行的方法放在 activated 中;

基本用法

<!--被keepalive包含的组件会被缓存-->
<keep-alive>
  <component><component />
</keep-alive>

被keepalive包含的组件不会被再次初始化,也就意味着不会重走生命周期函数
但是有时候是希望我们缓存的组件可以能够再次进行渲染,这时 Vue 为我们解决了这个问题
被包含在 keep-alive 中创建的组件,会多出两个生命周期的钩子: activated 与 deactivated:

  • activated 当 keepalive 包含的组件再次渲染的时候触发
  • deactivated 当 keepalive 包含的组件销毁的时候触发

keepalive是一个抽象的组件,缓存的组件不会被 mounted,为此提供activated和deactivated钩子函数

参数理解

keepalive 可以接收3个属性做为参数进行匹配对应的组件进行缓存:

  • include 包含的组件(可以为字符串,数组,以及正则表达式,只有匹配的组件会被缓存)
  • exclude 排除的组件(以为字符串,数组,以及正则表达式,任何匹配的组件都不会被缓存)
  • max 缓存组件的最大值(类型为字符或者数字,可以控制缓存组件的个数)

注:当使用正则表达式或者数组时,一定要使用 v-bind

<!-- 将(只)缓存组件name为a或者b的组件, 结合动态组件使用 -->
<keep-alive include="a,b">
 <component></component>
</keep-alive>

<!-- 组件name为c的组件不缓存(可以保留它的状态或避免重新渲染) -->
<keep-alive exclude="c"> 
 <component></component>
</keep-alive>

<!-- 使用正则表达式,需使用v-bind -->
<keep-alive :include="/a|b/">
 <component :is="view"></component>
</keep-alive>

<!-- 动态判断 -->
<keep-alive :include="includedComponents">
 <router-view></router-view>
</keep-alive>

<!-- 如果同时使用include,exclude,那么exclude优先于include, 下面的例子只缓存a组件 -->
<keep-alive include="a,b" exclude="b"> 
 <component></component>
</keep-alive>

<!-- 如果缓存的组件超过了max设定的值5,那么将删除第一个缓存的组件 -->
<keep-alive exclude="c" max="5"> 
 <component></component>
</keep-alive>

遇见 vue-router 结合router使用,缓存部分页面

所有路径下的视图组件都会被缓存

<keep-alive>
  <router-view>
    <!-- 所有路径匹配到的视图组件都会被缓存! -->
  </router-view>
</keep-alive>

如果只想要router-view里面的某个组件被缓存,怎么办?

  • 使用 include/exclude
  • 使用 meta 属性

1、用 include (exclude例子类似)

缺点:需要知道组件的 name,项目复杂的时候不是很好的选择

<keep-alive include="a">
  <router-view>
    <!-- 只有路径匹配到的 include 为 a 组件会被缓存 -->
  </router-view>
</keep-alive>

2、使用 meta 属性

优点:不需要例举出需要被缓存组件名称

使用$route.meta的keepAlive属性:

<keep-alive>
  <router-view v-if="$route.meta.keepAlive"></router-view>
</keep-alive>
<router-view v-if="!$route.meta.keepAlive"></router-view>

需要在router中设置router的元信息meta:

//...router.js
export default new Router({
 routes: [
  {
   path: '/',
   name: 'Hello',
   component: Hello,
   meta: {
    keepAlive: false // 不需要缓存
   }
  },
  {
   path: '/page1',
   name: 'Page1',
   component: Page1,
   meta: {
    keepAlive: true // 需要被缓存
   }
  }
 ]
})

【加盐】使用 router.meta 拓展

假设这里有 3 个路由: A、B、C。

需求:

  • 默认显示 A
  • B 跳到 A,A 不刷新
  • C 跳到 A,A 刷新

实现方式

在 A 路由里面设置 meta 属性:

{
    path: '/',
    name: 'A',
    component: A,
    meta: {
      keepAlive: true // 需要被缓存
    }
}

在 B 组件里面设置 beforeRouteLeave:

export default {
    data() {
      return {};
    },
    methods: {},
    beforeRouteLeave(to, from, next) {
       // 设置下一个路由的 meta
      to.meta.keepAlive = true; // 让 A 缓存,即不刷新
      next();
    }
};

在 C 组件里面设置 beforeRouteLeave:

export default {
    data() {
      return {};
    },
    methods: {},
    beforeRouteLeave(to, from, next) {
      // 设置下一个路由的 meta
      to.meta.keepAlive = false; // 让 A 不缓存,即刷新
      next();
    }
};

这样便能实现 B 回到 A,A 不刷新;而 C 回到 A 则刷新。

防坑指南

1.keep-alive 先匹配被包含组件的 name 字段,如果 name 不可用,则匹配当前组件 components 配置中的注册名称。
2.keep-alive 不会在函数式组件中正常工作,因为它们没有缓存实例。
3.当匹配条件同时在 include 与 exclude 存在时,以 exclude 优先级最高(当前vue 2.4.2 version)。比如:包含于排除同时匹配到了组件A,那组件A不会被缓存。
4.包含在 keep-alive 中,但符合 exclude ,不会调用 activated 和 deactivated。

实现前进刷新,后退不刷新

感谢 iceuncle 分享的 《vue实现前进刷新,后退不刷新》。

总结

路由大法不错,不需要关心哪个页面跳转过来的,只要 router.go(-1) 就能回去,不需要额外参数。

在非单页应用的时候,keep-alive 并不能有效的缓存了= =

keep-alive生命周期钩子函数:activated、deactivated

使用<keep-alive>会将数据保留在内存中,如果要在每次进入页面的时候获取最新的数据,需要在activated阶段获取数据,承担原来created钩子中获取数据的任务。

附录

生命周期函数:就是vue在某个时间段会自动执行的函数

  • beforeCreate(){}在执行的时候,data还有methods都没有被初始化
  • created(){} data还有methods都被初始化好了,如果要调用 methods 方法或者操作 data 里面的数据,最早只能在 created 里面进行操作。
  • beforeMount(){} 表示模板已经在内存中编辑完成了,但是尚未渲染到模板页面中。即页面中的元素,没有被真正的替换过来,只是之前写的一些模板字符串。
  • mounted(){} 表示内存中模板已经真实的挂载到页面中去了,用户可以看到渲染好的界面了

注意这是一个生命周期函数的最后一个函数了,执行完这个函数表示 整个vue实例已经初始化完成了,组件脱离了创建阶段,进入运行阶段。

下面是运行期间的两个生命周期函数的钩子:

beforeUpdate(){} 表示我们的界面还没更新 但是data里面的数据是最新的。即页面尚未和最新的data里面的数据保持同步。

update(){} 表示页面和data里面的数据已经包吃同步了 都是最新的。

beforeDestory(){} 当执行这个生命周期钩子的时候 vue的实例从运行阶段进入销毁阶段 此时实例身上的data 还有 methods处于可用的状态。

Destoryed(){} 表示组件已经完全被销毁了 组件中所有的实例方法都是不能用了

参考:

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

Javascript 相关文章推荐
避免 showModalDialog 弹出新窗体的原因分析
May 31 Javascript
js替换字符串的所有示例代码
Jul 23 Javascript
深入理解javascript中return的作用
Dec 30 Javascript
js 获取浏览器版本以此来调整CSS的样式
Jun 03 Javascript
简介JavaScript中用于处理正切的Math.tan()方法
Jun 15 Javascript
bootstrap实现弹窗和拖动效果
Jan 03 Javascript
详述JavaScript实现继承的几种方式(推荐)
Mar 22 Javascript
去除html代码里面的script正则方法
May 19 Javascript
详解AngularJS2 Http服务
Jun 26 Javascript
Vue 组件(component)教程之实现精美的日历方法示例
Jan 08 Javascript
js利用拖放实现添加删除
Aug 27 Javascript
js实现简易计算器小功能
Nov 18 Javascript
vue element 生成无线级左侧菜单的实现代码
Aug 21 #Javascript
微信小程序仿今日头条导航栏滚动解析
Aug 20 #Javascript
Vue中axios的封装(报错、鉴权、跳转、拦截、提示)
Aug 20 #Javascript
Vue formData实现图片上传
Aug 20 #Javascript
Angular8 Http拦截器简单使用教程
Aug 20 #Javascript
vue实现axios图片上传功能
Aug 20 #Javascript
扫微信小程序码实现网站登陆实现解析
Aug 20 #Javascript
You might like
一键删除顽固的空文件夹 软件下载
2007/01/26 PHP
分页详解 从此分页无忧(PHP+mysql)
2007/11/23 PHP
PHP实现今天是星期几的几种写法
2013/09/26 PHP
php使用json_encode对变量json编码
2014/04/07 PHP
PHP制作登录异常ip检测功能的实例代码
2016/11/16 PHP
php观察者模式应用场景实例详解
2017/02/03 PHP
php使用curl实现简单模拟提交表单功能
2017/05/15 PHP
JavaScript高级程序设计 学习笔记 js高级技巧
2011/09/20 Javascript
jBox 2.3基于jquery的最新多功能对话框插件 常见使用问题解答
2011/11/10 Javascript
瀑布流布局并自动加载实现代码
2013/03/12 Javascript
JavaScript获取客户端IP的方法(新方法)
2016/03/11 Javascript
ionic隐藏tabs的方法
2016/08/29 Javascript
基于jquery实现多选下拉列表
2017/08/02 jQuery
setTimeout时间设置为0详细解析
2018/03/13 Javascript
教你如何编写Vue.js的单元测试的方法
2018/10/17 Javascript
nodejs中函数的调用实例详解
2018/10/31 NodeJs
Node.js 实现简单的无侵入式缓存框架的方法
2019/07/21 Javascript
JavaScript实现图片上传并预览并提交ajax
2019/09/30 Javascript
vue用ant design中table表格,点击某行时触发的事件操作
2020/10/28 Javascript
老生常谈Python基础之字符编码
2017/06/14 Python
python链接oracle数据库以及数据库的增删改查实例
2018/01/30 Python
Python DataFrame.groupby()聚合函数,分组级运算
2018/09/18 Python
Python将字符串常量转化为变量方法总结
2019/03/17 Python
python opencv 读取图片 返回图片某像素点的b,g,r值的实现方法
2019/07/03 Python
python实现输出一个序列的所有子序列示例
2019/11/18 Python
为什么相对PHP黑python的更少
2020/06/21 Python
英国最大的高品质珠宝和手表专家:Goldsmiths
2017/03/11 全球购物
Europcar葡萄牙:葡萄牙汽车和货车租赁
2017/10/13 全球购物
Under Armour安德玛英国官网:美国高端运动科技品牌
2018/09/17 全球购物
考试不及格的检讨书
2014/01/22 职场文书
生物技术专业求职信
2014/06/10 职场文书
保证书格式
2015/01/16 职场文书
小学生2015教师节演讲稿
2015/03/19 职场文书
OpenCV中resize函数插值算法的实现过程(五种)
2021/06/05 Python
通过Qt连接OpenGauss数据库的详细教程
2021/06/23 PostgreSQL
Mybatis-Plus进阶分页与乐观锁插件及通用枚举和多数据源详解
2022/03/21 Java/Android