Vue缓存组件或页面实用技巧 - keepAlive销毁

简介: 在Vue中,我们可以很方便的使用 keep-alive 来缓存页面,但你知道它存在什么坑吗?

假设在一个列表中,用户滑动几页点击了详情,此时若再回到列表页,页面状态都已经刷新,用户又需要再进行滑动,这显然是不合理的。

在PC端我们有很多天然优势可以处理这类问题,比如使用分页器让列表不要过长,比如打开详情跳转个新标签页,又或者将详情页面做成抽屉,做成遮罩弹窗等等。。但是在移动端开发中,我们无法避免这个问题,所以有时候就必须记录下页面的状态,让用户在返回时保持页面原有的样子。

在Vue中,我们可以很方便的使用keep-alive来缓存页面,通常情况下,我们可以通过如下方式改写路由视图:

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

如上使用了两个if判断,不然会不生效

<!-- 错误写法: -->
<keep-alive v-if="$route.meta.keepAlive">
    <router-view></router-view>
</keep-alive>
    <router-view v-else></router-view>

然后我们在需要缓存的页面路由处设置meta即可:

{
    path: "/activityList",
    name: "activityList",
    title: "活动列表",
    meta: { keepAlive: true },
    component: () => import("@/pages/activity/list.vue")
}

这种就是最常用的方式,在使用了keepAlive之后的页面,生命周期会发生一些变化,比如create和mounted这种页面初始化相关的钩子只会执行一次,因为页面的状态已经不会自动改变了,同时有两个额外的生命周期钩子会生效,分别是:

activated() {
    // 页面重回
},
deactivated() {
    // 页面离开
},

除此之外使用了keepAlive还有一个副作用,我们会发现页面无法卸载了,也就是销毁的生命周期不再触发了,而且当用户离开列表页,转而又进入列表页的时候,列表页应该是初始化状态而不是缓存状态等情况,这些都使得业务逻辑代码变得繁杂。

为了解决以上的问题,我们采用Vue提供的include方式来替代if判断

    <keep-alive :include="keepPages">
        <router-view></router-view>
    </keep-alive>

并且结合Vuex来管理状态

  computed: {
    keepPages() {
      return this.$store.getters.getKeepPages
    }
  }

mutation代码:

  changeKeepPages(state, status) {
    state.keepPages = status;
  },
  keepThisPage(state, status) {
    const arr = state.keepPages.split(",");
    arr.push(status);
    state.keepPages = arr + "";
  },
  removeThisPage(state, status) {
    const arr = state.keepPages.split(",");
    const index = arr.findIndex(x => x === status);
    if (index > 0) {
      arr.splice(index, 1);
    }
    state.keepPages = arr + "";
  }

然后又写了一份mixin,混入即可将该页面/组件设置为缓存状态:

export default {
  created() {
    if (!this.$options.name) { // 这里的name是页面/组件对象中的name,不是路由的
      console.warn("缓存页面失败!没有设置组件名!");
    }
    this.$store.commit("keepThisPage", this.$options.name);
  },
  beforeDestroy() {
    console.log("触发组件销毁")
  }
};

销毁页面可以主动使用:

this.$store.commit('removeKeepPages', 'xxx') // xxx为要销毁的页面名称

也可以清空所有缓存的页面,比如在用户回到首页的时候触发:

this.$store.commit('changeKeepPages', 'index') // index这个地方没有实际意义,经测试include为空值时所有页面都会缓存,所以这里才设置了一个不存在的。

我这里使用了下面的方式,在回到首页时顺利触发了前一个列表页的销毁事件,这样整个流程就变得十分舒适了,也在一定程度上避免了内存泄漏。以上就是Vue中keepAlive的实用技巧,希望对你有所帮助。

相关文章
|
9天前
|
存储 JavaScript 开发者
Vue 组件间通信的最佳实践
本文总结了 Vue.js 中组件间通信的多种方法,包括 props、事件、Vuex 状态管理等,帮助开发者选择最适合项目需求的通信方式,提高开发效率和代码可维护性。
|
9天前
|
存储 JavaScript
Vue 组件间如何通信
Vue组件间通信是指在Vue应用中,不同组件之间传递数据和事件的方法。常用的方式有:props、自定义事件、$emit、$attrs、$refs、provide/inject、Vuex等。掌握这些方法可以实现父子组件、兄弟组件及跨级组件间的高效通信。
|
20天前
|
缓存 JavaScript UED
Vue 中实现组件的懒加载
【10月更文挑战第23天】组件的懒加载是 Vue 应用中提高性能的重要手段之一。通过合理运用动态导入、路由配置等方式,可以实现组件的按需加载,减少资源浪费,提高应用的响应速度和用户体验。在实际应用中,需要根据具体情况选择合适的懒加载方式,并结合性能优化的其他措施,以打造更高效、更优质的 Vue 应用。
|
25天前
|
前端开发 UED
vue3知识点:Suspense组件
vue3知识点:Suspense组件
30 4
|
24天前
|
JavaScript 前端开发 测试技术
组件化开发:创建可重用的Vue组件
【10月更文挑战第21天】组件化开发:创建可重用的Vue组件
24 1
|
25天前
|
JavaScript 前端开发 Java
《vue3第五章》新的组件,包含:Fragment、Teleport、Suspense
《vue3第五章》新的组件,包含:Fragment、Teleport、Suspense
32 2
|
25天前
|
Java
vue3知识点:Teleport组件
vue3知识点:Teleport组件
25 1
|
21天前
|
JavaScript UED
"Vue实战技巧大揭秘:一招解决路由跳转页面不回顶部难题,让你的单页面应用用户体验飙升!"
【10月更文挑战第23天】在Vue单页面应用中,点击路由跳转时,默认情况下页面不会自动滚动到顶部,这可能影响用户体验。本文通过一个新闻网站的案例,介绍了如何使用Vue-router的全局前置守卫和`scrollBehavior`方法,实现路由跳转时页面自动滚动到顶部的功能,提升用户浏览体验。
56 0
|
1月前
|
消息中间件 缓存 NoSQL
Redis 是一个高性能的键值对存储系统,常用于缓存、消息队列和会话管理等场景。
【10月更文挑战第4天】Redis 是一个高性能的键值对存储系统,常用于缓存、消息队列和会话管理等场景。随着数据增长,有时需要将 Redis 数据导出以进行分析、备份或迁移。本文详细介绍几种导出方法:1)使用 Redis 命令与重定向;2)利用 Redis 的 RDB 和 AOF 持久化功能;3)借助第三方工具如 `redis-dump`。每种方法均附有示例代码,帮助你轻松完成数据导出任务。无论数据量大小,总有一款适合你。
74 6
|
8天前
|
缓存 NoSQL 关系型数据库
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题
本文详解缓存雪崩、缓存穿透、缓存并发及缓存预热等问题,提供高可用解决方案,帮助你在大厂面试和实际工作中应对这些常见并发场景。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题