写在前面
一个web app的实际使用场景中,有一些情景的交互要求,是记录用户的浏览状态的。最常见的就是在列表页进入详情页之后,再返回到列表页,用户希望返回到进入详情页之前的状态继续操作。但是有些使用场景,用户又是希望能够获取最新的数据,例如同级列表页之间切换的时候。
如此,针对上述两种使用场景,需要实现按需读取页面缓存。由于SPA应用的路由逻辑也是在前端实现的,因此可以在前端对路由的逻辑进行设置以实现所需效果。
使用技术
- Vue.js作为主要框架
- Vue-router作为前端路由管理器
- Vuex作为状态管理工具
总体思路
keep-alive判断当前组件是否读取缓存的节点,在整个生命周期里面非常靠后,在afterEach之后,基本在组件实例创建之前。(因此在此之前对当前组件是否读取缓存进行处理都是可行的,我选择在全局前置守卫进行处理)
而判断当前组件是否缓存的节点,则早于组件的beforeRouteLeave钩子。
基于上述逻辑,本方案解决的逻辑是,对当前打开的页面进行判断,动态生成需要keepAlive的组件数组配置,对有可能需要缓存的先行进行缓存,然后在每次路由切换的时候,再进行判断,按需读取页面缓存。
- 使用kepp-alive进行缓存,使用include属性对需要缓存的页面进行配置。
- 由于需要缓存的页面配置系动态生成,所以使用vuex储存该配置。
- 在路由元信息中写入两个配置,一是该路由是否需要缓存,二是从相关路由进入时才进行缓存的特定路由数组。
- 在beforeEach进行设置,每次进入路由之前,对进入的路由及其所有父级路由进行判断,若需要缓存且命中特定路由数组,则将相关路由添加至缓存配置文件中;若不符合,则将相关路由删除。(此步骤实现了路由切换时,需要则读取缓存,不需要则重新获取数据。)
- 使用全局mixin,进入相关组件之前,对当前路由进行判断,如果需要缓存的则将该路由添加至缓存配置中。(此步骤实现了缓存当前打开的需要缓存的页面。)
具体实现
1. 使用include属性控制路由缓存
此处需要注意的是,include匹配首先检查组件自身的 name 选项,如果 name 选项不可用,则匹配它的局部注册名称 (父组件 components 选项的键值)。匿名组件不能被匹配。
但是vue-router的环境下,是没有局部注册名称的,只能为组件补全name属性。
因此,请务必给组件添加 name 选项,否则匿名组件将全部应用缓存。
<keep-alive :include="$store.state.cachedRouteNames"> <router-view /> </keep-alive>
2. 添加全局路由缓存配置
// store/index.js const store = new vuex.Store({ state: { // 缓存的路由列表 cachedRouteNames: [], }, mutations: { UPDATE_CACHEDROUTENAMES(state,{ action, route }) { const methods = { 'add': () => { state.cachedRouteNames.push(route) }, 'delete': () => { state.cachedRouteNames.splice(state.cachedRouteNames.findIndex((e) => { return e === route}),1) } } methods[action]() } } })
3. 配置路由元信息,对需要缓存的路由进行配置
keepAlive表明路由需要被缓存,必须,否则不缓存
cacheWhenFromRoutes为数组,非必须,若为falsy值,则任何时候均缓存;若为空数组,则任何时候均不缓存
// router/index.js { path: '/productslist', name: 'ProductsList', component: ProductsList, meta: { keepAlive: true, cacheWhenFromRoutes: ['ProductDetail'] // 此处配置的是路由的name } },
4. 配置全局前置守卫,按需读取缓存
// routeControl.js // 需要缓存的路由名称数组 const cachedRouteNames = store.state.cachedRouteNames; // 定义添加缓存组件name函数,设置的是组件的name const addRoutes = (route) => { const routeName = route.components.default.name if (routeName && cachedRouteNames.indexOf(routeName) === -1) { store.commit('UPDATE_CACHEDROUTENAMES', { action: 'add', route: routeName }) } } // 定义删除缓存组件name函数,设置的是组件的name const deleteRoutes = (route) => { const routeName = route.components.default.name if (routeName && cachedRouteNames.indexOf(routeName) !== -1) { store.commit('UPDATE_CACHEDROUTENAMES', { action: 'delete', route: routeName }) } } router.beforeEach((to, from, next) => { // 处理缓存路由开始 // 在读取缓存之前,先对该组件是否读取缓存进行处理 to.matched.forEach((item, index) => { const routes = item.meta.cacheWhenFromRoutes; /** * 此处有几种情况 * 1. 没有配置cacheWhenFromRoutes, 则一直缓存; * 2. 配置了cacheWhenFromRoutes,但是首次打开此web app,则from.name为空,此时应该将该页面组件的name添加到缓存配置文件中 * 3. 配置了cacheWhenFromRoutes,from.name不为空,若命中cacheWhenFromRoutes,则添加该页面组件的name到缓存配置文件中,否则删除。 * **/ if (item.meta.keepAlive && (!routes || (routes && (!from.name || routes.indexOf(from.name) !== -1)))) { addRoutes(item) } else { deleteRoutes(item) } }) // 处理缓存路由结束 new Promise(( resolve, reject ) => { // ..other codes }).then( res => { if ( res ) { next(res) } else { next() } }) }) // 全局混入。此步骤的目的是在该组件被解析之后,若是属于需要缓存的组件,先将其添加到缓存配置中,进行缓存。 // 导航守卫的最后一个步骤就是调用 beforeRouteEnter 守卫中传给 next 的回调函数,此时整个组件已经被解析,DOM也已经更新。 Vue.mixin({ beforeRouteEnter(to, from, next) { next(vm => { to.matched.forEach((item) => { const routeName = item.components.default.name if (to.meta.keepAlive && routeName && cachedRouteNames.indexOf(routeName) === -1) { store.commit('UPDATE_CACHEDROUTENAMES', { action: 'add', route: routeName }) } }) }) }, })
写在最后
坑点
- 此方案涉及两个name,一个是设置特定路由时,使用路由的name。另一个是动态生成缓存配置文件时,使用的是页面组件的name。
- 务必给组件添加name属性,便于include属性的使用,也方便调试跟踪。如果组件缺少name属性,将会默认使用缓存。
- 动态处理缓存配置时,一定要对to.matched进行遍历,否则嵌套路由的父级路由的缓存就无法生效,将导致子路由的缓存也无法生效。
- 全局混入有一定危险性,慎用...
以上是实践过程中摸索出来的一种解决方案,我相信存在更加优雅高效的解决方式。如果你正好实践过相关方法,烦请指正,谢谢。
更多参考
github.com/vuejs/vue/i…
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
稳了!魔兽国服回归的3条重磅消息!官宣时间再确认!
昨天有一位朋友在大神群里分享,自己亚服账号被封号之后居然弹出了国服的封号信息对话框。
这里面让他访问的是一个国服的战网网址,com.cn和后面的zh都非常明白地表明这就是国服战网。
而他在复制这个网址并且进行登录之后,确实是网易的网址,也就是我们熟悉的停服之后国服发布的暴雪游戏产品运营到期开放退款的说明。这是一件比较奇怪的事情,因为以前都没有出现这样的情况,现在突然提示跳转到国服战网的网址,是不是说明了简体中文客户端已经开始进行更新了呢?
更新日志
- 凤飞飞《我们的主题曲》飞跃制作[正版原抓WAV+CUE]
- 刘嘉亮《亮情歌2》[WAV+CUE][1G]
- 红馆40·谭咏麟《歌者恋歌浓情30年演唱会》3CD[低速原抓WAV+CUE][1.8G]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[320K/MP3][193.25MB]
- 【轻音乐】曼托凡尼乐团《精选辑》2CD.1998[FLAC+CUE整轨]
- 邝美云《心中有爱》1989年香港DMIJP版1MTO东芝首版[WAV+CUE]
- 群星《情叹-发烧女声DSD》天籁女声发烧碟[WAV+CUE]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[FLAC/分轨][748.03MB]
- 理想混蛋《Origin Sessions》[320K/MP3][37.47MB]
- 公馆青少年《我其实一点都不酷》[320K/MP3][78.78MB]
- 群星《情叹-发烧男声DSD》最值得珍藏的完美男声[WAV+CUE]
- 群星《国韵飘香·贵妃醉酒HQCD黑胶王》2CD[WAV]
- 卫兰《DAUGHTER》【低速原抓WAV+CUE】
- 公馆青少年《我其实一点都不酷》[FLAC/分轨][398.22MB]
- ZWEI《迟暮的花 (Explicit)》[320K/MP3][57.16MB]