填坑小能手——页面缓存

前言

最近项目需求需要将页面进行缓存,这个需求直接用keep-alive就能实现。当时预估着很快就能完成需求,结果还是踩了一些的坑,大家将来如果有类似需求可以避免一下。

keep-alive

keep-alivevue的内置组件,不是vue-router的。使用keep-alive能够将当前页面进行缓存,再次进入页面不会触发页面mounted等生命周期,为了满足需要keep-alive有两个单独的生命周期:

js 复制代码
activated() {
    //加载时调用
  },
  deactivated() {
    //卸载后调用
  }

keep-alive的用法也很简单,将需要缓存的组件进行包裹即可,一般会跟router-view配合,包裹的router-view最好加个key值,路由地址对应的组件可能是同一个。

js 复制代码
<keep-alive>
 <router-view :key="key" />  //key可以为name或者pullPath
</keep-alive>

默认情况下keep-alive会缓存所有的组件,为了更加灵活的使用我们可以使用相关参数include与exclude参数。

js 复制代码
// include 包含 exclude排除
<keep-alive :include=cachedViews :exclude=notCached>
 <router-view :key="key" />
</keep-alive>

注意,接受的参数是组件name的集合 。实际开发中缓存一般是可控的,我们可以在路由配置项的meta中增加一个属性控制。

js 复制代码
{
path: '/home',
component: Home,
meta: { noCache: true }
},

cacheViews参数的值需要过滤掉noCache为true的值。

缓存不生效

前期工作准备完善,满怀期待打开页面,然后发现缓存不生效。网上查了查,缓存不生效的情况有以下几种:

  1. keep-alive位置放错,缓存的页面没被包裹进去。
  2. include参数里的name与组件名称没对齐。这里的name是组件名称,而不是路由名称
  3. 嵌套路由,这种算是第二种的特殊情况,我们会重点讲解下。

知道了原因,改起来就很快了。简单排查了下,发现项目中cachedViews添加的是路由名称:

js 复制代码
if (view.name === null) return
if (this.cachedViews.includes(view.name)) return
if (!view.meta.noCache) {
this.cachedViews.push(view.name)
}

再点开对应组件,发现没有name属性,难怪不生效,这就是第二种情况了,添加name属性为路由名称,再试下可以了。当我以为万事大吉的时候,发现某些页面还不生效,而这些页面都是有二级菜单也就是嵌套路由,嵌套路由多了一个router-veiw,这是个公共组件:

根据上面的原则,我把cachedViews中添加了'CommonRouterView',缓存生效了。但是这样子做所有子路由都会被缓存,noCache属性也就没啥用了,还得优化下。

嵌套路由缓存

当时在掘金上搜索了下,找到了比较靠谱的解决方案,就是将嵌套路由扁平化 ,变成一维数组,icludes里面跟一维数组对比,嵌套路由也就能被缓存了,不过文章中没列举出源码,想了下也就没深究。后面看到可以从原理上入手,keep-alive的原理中会通过唯一的key将组件进行缓存,而这个key就是组件名称,多级路由情况下组件名称不能满足使用,我们可以将key与路由相关属性结合起来,而路由的fullPath具有唯一性真合适。 修改起来也很简单,将keep-alive的源码复制一份出来,然后将关键代码进行修改:

用新写的缓存组件包裹router-view,然后将cachedViews的取值进行修改。

js 复制代码
cachedViews() {
// 添加修改为fullPath
if (!view.meta.noCache) {
this.cachedViews.push(view.fullPath)
})
}

如此key值相对应,嵌套路由能够正确缓存并且noCache也能够控制。

总结

页面缓存看似简单,但想要页面正确缓存还是需要注意细节的,尤其是碰到多级路由的情况。

相关推荐
Moment13 分钟前
Vibe Coding 时代,到底该选什么样的工具来提升效率❓❓❓
前端·后端·github
IT_陈寒2 小时前
SpringBoot性能飙升200%?这5个隐藏配置你必须知道!
前端·人工智能·后端
小时前端2 小时前
React性能优化的完整方法论,附赠大厂面试通关技巧
前端·react.js
Nicko2 小时前
Jetpack Compose BOM 2026.02.01 解读与升级指南
前端
小蜜蜂dry3 小时前
nestjs学习 - 控制器、提供者、模块
前端·node.js·nestjs
优秀稳妥的JiaJi3 小时前
基于腾讯地图实现电子围栏绘制与校验
前端·vue.js·前端框架
前端开发呀3 小时前
从 qiankun(乾坤) 迁移到 Module Federation(模块联邦),对MF只能说相见恨晚!
前端
没想好d3 小时前
通用管理后台组件库-10-表单组件
前端
恋猫de小郭3 小时前
你用的 Claude 可能是虚假 Claude ,论文数据告诉你,Shadow API 中的欺骗性模型声明
前端·人工智能·ai编程
_Eleven4 小时前
Pinia vs Vuex 深度解析与完整实战指南
前端·javascript·vue.js