重庆分公司,新征程启航
为企业提供网站建设、域名注册、服务器等服务
这篇文章给大家分享的是有关vue里主动销毁keep-alive缓存的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。
创新互联建站主要从事成都网站制作、成都网站设计、网页设计、企业做网站、公司建网站等业务。立足成都服务阳江,10年网站建设经验,价格优惠、服务专业,欢迎来电咨询建站服务:028-86922220
问题产生的背景
我们一个后台,在切换一些标签页的时候,是使用的 keep-alive
缓存的标签页,也使用了 include
属性来决定哪个页面进行缓存,而标签页的切换实际上是路由的切换,也就是说打开一个新标签页的时候,url 会跟着变化,老的标签页如果在 keep-alive
的 include
范围内那就会缓存下来。
然后客服人员就反馈页面开的久了就会崩溃,因为他们基础上不会刷新页面(工作需要),又总有切换标签的习惯,最后导致内存越来越大最后崩溃。
依赖环境
这个项目是基于一个开源 vue
后台框架:https://github.com/PanJiaChen/vue-element-admin
,然后代码一直由几个后端开发维护的!所以后端没找出问题在哪,然后就我来接手这个问题了。
写文章时,标签里竟然没有 vue
这一项,差评!
定位问题
先梳理下业务逻辑:从业务场景来说,我们在标签页之间切换时,如果刚进入的这个标签页已被缓存了,那被缓存的标签页就直接拿出来展示就行,而关闭这个标签页的时候就应该销毁对应的组件。
花了点时间查看了一下代码,发现问题在于关闭标签页的时候,虽然这个页面没在 keep-alive
的 include
里了,但是组件也没有被销毁掉,还是在缓存状态,我们可以通过 Vue Devtools
插件看到关闭后的标签页对应的组件一直还存在着:
当然,在这块 keep-alive
本身的逻辑我觉得是没问题的,主要是我们项目比较复杂,缓存的组件太多了而且会一直增加,所以最终导致崩溃。
解决问题
既然问题已经定位了,那就好解决问题了,只需要在关闭标签页的时候把对应的组件也销毁掉就好了。
经过网上一翻查找,发现销毁一个组件可以使用: this.$destroy(‘组件名')
来销毁。
先说下大概思路:keep-alive
的 include
里存的其实是一个 vuex
中的一个数据源(数据源保存的是路由名称),当关闭标签页时,这个数据源中的一项会被移除。这之前,我们在组件里监听到这个数据源的变化,如果此组件对应的路由(这个路由应在 mounted
的时候保存下来)已经不在数据源中了,那就应该销毁此组件。
代码大概如下:
const mixin = { data() { return { routePath: '' } }, mounted() { this.routePath = this.$route.path }, computed: { visitedViews() { return this.$store.state.tagsView.visitedViews } }, watch: { visitedViews(value) { if(value 里没有了 routePath 这一项) this.$destroy(this.$options.name) } } } } export default mixin
这一段代码单独拎出来了,然后在需要缓存的组件里使用 mixins
混入到组件对象中,这样组件中要添加的代码量就比较少了。
更改后经过测试,关闭标签页后对应的组件就会被销毁掉,使用 Vue Devtools
能看的很清楚。
更多思考
在我们后台操作这么频繁的业务场景下,使用 keep-alive
其实并不是一个好的选择。
在我们修复这个问题后,我们通过控制台里的 Memory
查看页面内存的变化时,发现组件即便被销毁,也要经过一段时间才能回收完,当我们在这一段时间一直创建/打开新的标签页时,内存还是会在短时间内高涨。而且有时候,内存在经过一段时间后也并没有回收掉。
keep-alive
本质上是把整个 dom
节点及对应的事件等都缓存下来了,当这样的组件很多的时候,自然会占用很多内存。而如果我们只缓存这个组件中的数据,在需要这个组件再次显示的时候再临时渲染那肯定要节省很多内存的,毕竟数据占的空间其实很小的,而渲染组件要花的时间也不会很长(只要组件不是特别特别复杂)。
所以,下一阶段的优化工作就是把 keep-alive
去掉,然后使用 vuex
来缓存组件中的数据,当需要重新显示数据时再把数据取出来并重新渲染。当然,这是一个比较大的工程!
感谢各位的阅读!关于“vue里主动销毁keep-alive缓存”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!