Jamey's Jamey's
首页
导航站
  • 学习专栏

    • 《HTML》笔记
    • 《CSS》笔记
    • 《JavaScript》笔记
    • 《Vue》笔记
    • 《Git》笔记
    • 《规范》笔记
    • 《软技能》笔记
    • 《面试》笔记
    • 《持续集成&交付&部署》笔记
  • 踩坑专栏

    • 《Element-UI 实践系列》笔记
    • 《移动端 实践系列》笔记
    • 《综合》笔记
  • 配置专栏

    • 《环境系列》笔记
  • 极空间

    • Docker
  • 影视

    • movie
  • 编辑器笔记

    • 开发编辑器
  • 浏览器笔记

    • Chrome
  • Mac笔记

    • Mac
  • 跨界学习

    • 运营
  • 破解合集

    • 破解
  • 本站

    • 分类
    • 标签
    • 归档
  • 我的

    • 收藏
    • 书单
    • 关于

Jamey

首页
导航站
  • 学习专栏

    • 《HTML》笔记
    • 《CSS》笔记
    • 《JavaScript》笔记
    • 《Vue》笔记
    • 《Git》笔记
    • 《规范》笔记
    • 《软技能》笔记
    • 《面试》笔记
    • 《持续集成&交付&部署》笔记
  • 踩坑专栏

    • 《Element-UI 实践系列》笔记
    • 《移动端 实践系列》笔记
    • 《综合》笔记
  • 配置专栏

    • 《环境系列》笔记
  • 极空间

    • Docker
  • 影视

    • movie
  • 编辑器笔记

    • 开发编辑器
  • 浏览器笔记

    • Chrome
  • Mac笔记

    • Mac
  • 跨界学习

    • 运营
  • 破解合集

    • 破解
  • 本站

    • 分类
    • 标签
    • 归档
  • 我的

    • 收藏
    • 书单
    • 关于
  • 基础

  • 组件

  • 路由

    • keep-alive 深入理解及实践总结
      • 📖. 前言
      • 一. 什么是 keep-alive
      • 二. 声明周期执行顺序
      • 三. 基本用法
      • 四. 参数理解
      • 五. 缓存部分页面
        • 1. 全缓存
        • 2. 部分缓存
      • 六. 加盐:router.meta 拓展
      • 七. 防坑指南
      • 八. 总结
      • 九. 附录
      • 参考
  • 自定义指令

  • 实践踩坑

  • 知识点

  • 《Vue》笔记
  • 路由
Jamey
2019-08-19
目录

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

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

# 📖. 前言

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

# 一. 什么是 keep-alive

也就是说,keepalive 是 Vue 内置的一个组件,可以使被包含的组件保留状态,或避免重新渲染。也就是所谓的组件缓存

keep-alive 是 Vue 的内置组件,能在组件切换过程中将状态保留在内部中,房东重复渲染 DOM。

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

prop:

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

# 二. 声明周期执行顺序

  • 页面第一次进入,钩子的触发顺序: created → mounted → activated;退出时触发 deactivated;当再次进入(前进或者后退)时,只触发 activated
  • 事件挂载的方法等,只执行一次的放在 mounted 中;组件每次进去执行的方法放在 activated 中

# 三. 基本用法

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

被 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>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29

# 五. 缓存部分页面

结合 vue-router 和 router 使用,实现缓存部分页面效果。

# 1. 全缓存

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

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

# 2. 部分缓存

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

  • 使用 include / exclude
  • 使用 meta 属性
  1. 用 include(exclude 例子类似)

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

<keep-alive include="a">
  <router-view>
    <!-- 只有路径匹配到的 include 为 a 组件会被缓存 -->
  </router-view>
</keep-alive>
1
2
3
4
5
  1. 使用 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>
1
2
3
4
5

需要在 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 // 需要被缓存
      }
    }
  ]
})
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

# 六. 加盐:router.meta 拓展

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

  • 需求:

    • 默认显示 A
    • B 跳到 A,A 不刷新
    • C 调到 A,A 刷新
  • 实现方式

  1. 在 A 路由里面设置 meta 属性:
{
  path: '/',
  name: 'A',
  component: A,
  meta: {
    keepAlive: true // 需要被缓存
  }
}
1
2
3
4
5
6
7
8
  1. 在 B 组件里面设置 beforeRouteLeave:
export default {
  data() {
    return {};
  },
  methods: {},
  beforeRouteLeave(to, from, next) {
    // 设置下一个路由的 meta
    to.meta.keepAlive = true;  // 让 A 缓存,即不刷新
    next();
  }
};
1
2
3
4
5
6
7
8
9
10
11
  1. 在 C 组件里面设置 beforeRouteLeave:
export default {
  data() {
    return {};
  },
  methods: {},
  beforeRouteLeave(to, from, next) {
    // 设置下一个路由的 meta
    to.meta.keepAlive = false; // 让 A 不缓存,即刷新
    next();
  }
};
1
2
3
4
5
6
7
8
9
10
11

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

# 七. 防坑指南

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

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

# 八. 总结

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

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

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

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

# 九. 附录

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

  1. beforeCreate() {} → 在执行的时候,data 还有 methods 都还没有被初始化。
  2. created() {} → data 还有 methods 都被初始化好了,如果要调用 methods 方法或者操作 data 里面的数据,最早只能在 created 里面进行操作。
  3. beforeMount() {} → 表示模板已经在内存中编译完成了,但是尚未渲染到模板页面中。即页面中的元素,没有被真正的替换过来,只是之前写的一些模板字符串。
  4. mounted() {} → 表示内存中模板已经真实的挂载到页面中去了,用户可以看到渲染好的界面了。
    • 注意:这是生命周期函数的最后一个函数了,执行完这个函数表示:整个 vue 实例已经初始化完成了,组件脱离了创建阶段,进入运行阶段。
    • 下面是运行期间的两个生命周期函数的钩子:
  5. beforeUpdate() {} → 表示我们的界面还没更新,但是 data 里面的数据是最新的。即页面尚未和最新的 data 里面的数据保持同步。
  6. updated() {} → 表示页面和 data 里面的数据已经保持同步了,都是最新的。
  7. beforeDestory() {} → 当执行这个生命周期钩子的时候,vue 的实例从运行阶段进入销毁阶段。此时实例身上的 data 还有 methods 处于可用的状态。
  8. destoryed() {} → 表示组件已经完全销毁了,组件中所有的实例方法都不能用了。

# 参考

  • issues#811 (opens new window)
  • vue#keep-alive (opens new window)
  • vue2.0 keep-alive最佳实践 (opens new window)
#Vue.js2
上次更新: 2022/07/08, 18:18:02
计算属性怎么传参?
自定义指令介绍

← 计算属性怎么传参? 自定义指令介绍→

Theme by Vdoing | Copyright © 2017-2023 Jamey | blog 闽ICP备19022664号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式