黑马程序员技术交流社区

标题: 【上海校区】Vue 全站缓存之 keep-alive : 动态移除缓存 [打印本页]

作者: 不二晨    时间: 2018-8-3 10:32
标题: 【上海校区】Vue 全站缓存之 keep-alive : 动态移除缓存
前言以一个记账项目举例,常见的场景有首页、记到账页面、选择合同、新建合同、选择客户、新建客户这些页面。


在这些页面中,很显然,用户的浏览行为应该是逐渐深入的,通俗得讲就是浏览页面在不断前进。
而且这些页面之间还是有互动性存在的,两种互动行为:


上图是 demo 项目中的真实效果,目前常见的 vue 开发方案里,一般都会引入 vuex 或 localStorage ,在各个页面不断的存储和调用页面内的数据,我觉得,这很不科学很不优雅
keep-alive 什么问题vue 支持 keep-alive 组件,如果启用,页面内的所有数据都会被保留,所以,上文的互动行为二后退时保留前一页数据继续操作没有问题。
问题出在互动行为一用户前进时总是进入新页面,然而一旦缓存,你就没法总是进新页面了,你总是进入缓存页,这就很让人头疼了。
官方提供了include和exclude特性,说你可以决定哪些页面使用缓存哪些页面不用缓存。链接
然而问题又回到了原点,并没有解决我们酌情决定是否使用已缓存的缓存这一需求。
所以很多人想到了一个方法在离开页面时销毁这个页面是不是就可以了,然而并不能,这里出现了 bug ,组件销毁了缓存还在:


于是,就有人提出希望keep-alive能增加可以动态删除已缓存组件的功能,issue
这是个老话题,之前一直没有进展,核心原因就在于 keep-alive 不能正确处理已销毁的组件。
尝试解决这个问题如果能实现动态使用缓存这一功能,那么所有问题也就迎刃而解。
最初,我研究 keep-alive 的代码,发现了这么一段代码:


于是,我想,如果在此处判断如果组件已被销毁则不使用缓存,是不是就解决这个问题了,于是我提交了一个 PR:


不过这个 PR 迟迟没有通过,我就放弃了。
暴力解决这个问题我继续研究有没有其他方案,然后我在打印组件变量的时候,发现了这么个眼熟的字段:


这不就是 keep-alive 的组件嘛,我赶忙点开再看,发现了更眼熟的东东:


于是,这事儿就变得简单了,直接按图索骥,咱在销毁组件之前,寻找路由组件所在父级的 keep-alive 组件,操控其中的 cache 列表,强行删除其中的缓存,问题也就迎刃而解,是不是很直接很暴力。
结论keep-alive 默认不支持动态销毁已缓存的组件,所以此处给出的解决方案是通过直接操控 keep-alvie 组件里的 cahce 列表,暴力移除缓存:
//使用Vue.mixin的方法拦截了路由离开事件,并在该拦截方法中实现了销毁页面缓存的功能。Vue.mixin({    beforeRouteLeave:function(to, from, next){        if (from && from.meta.rank && to.meta.rank && from.meta.rank>to.meta.rank)        {//此处判断是如果返回上一层,你可以根据自己的业务更改此处的判断逻辑,酌情决定是否摧毁本层缓存。            if (this.$vnode && this.$vnode.data.keepAlive)            {                if (this.$vnode.parent && this.$vnode.parent.componentInstance && this.$vnode.parent.componentInstance.cache)                {                    if (this.$vnode.componentOptions)                    {                        var key = this.$vnode.key == null                                    ? this.$vnode.componentOptions.Ctor.cid + (this.$vnode.componentOptions.tag ? `::${this.$vnode.componentOptions.tag}` : '')                                    : this.$vnode.key;                        var cache = this.$vnode.parent.componentInstance.cache;                        var keys  = this.$vnode.parent.componentInstance.keys;                        if (cache[key])                        {                            if (keys.length) {                                var index = keys.indexOf(key);                                if (index > -1) {                                    keys.splice(index, 1);                                }                            }                            delete cache[key];                        }                    }                }            }            this.$destroy();        }        next();    },});复制代码后语本文主要围绕如何动态删除 keep-alive 缓存这一问题进行探索,其他关于如何设定页面层级、如何在前后页之间进行数据传递等问题,敬请期待《Vue 全站缓存之 vue-router-then :前后页数据传递》。


作者:wanyaxing
链接:https://juejin.im/post/5b610da4e51d45195c07720d




作者: 不二晨    时间: 2018-8-16 17:25
奈斯




欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2