详解Vue依赖收集引发的问题


Posted in Javascript onApril 22, 2019

问题背景

在我们的项目中有一个可视化配置的模块,是通过go.js生成canvas来实现的。但是,我们发现这个模块在浏览器中经常会引起该tab页崩溃。开启chrome的任务管理器一看,进入该页面内存和cpu就会暴涨,内存经常会飙到700多M。但是我们的canvas实际的像素只有约500x500,根据一些粗略的计算,大概只占了1M的内存,这个计算过程可参考100*100的 canvas 占多少内存。那么我们这700M内存是哪里来的呢?

定位问题

我们可以使用chrome开发者工具来分析我们的调用栈。这边我是先通过Performance来帮助我们定位问题,它会帮我们生成一段过程中一些数据的变化,包括js堆内存、dom节点数量、动画帧等数据,如图:

详解Vue依赖收集引发的问题

这是切换至一个canvas画布较大的一个模块的performance分析表现,可以看到占用了472M的内存。下面折线图蓝色部分是js堆内存的变化,而Main下面黄色与紫色的矩形框就是我们的调用栈,上下两部分是按照时间一一对应的。可以看到,蓝色的折线呈高低起伏的态势,GC回收之后低点基本和高点持平,因此可以断定几乎不存在内存泄漏的问题。然后我们可以放大去看一看,内存升高的时候,js做了些什么事情,找一找规律。

 详解Vue依赖收集引发的问题

我们随机找一段内存增长的区域,可以看到在内存增长的过程中,最为频繁调用的就是Observer相关的代码。但是就这么看,我们不能够明白Observer是在干什么。此时我们可以借用Memory选项中的Allocation Sampling按照javascript function来查看内存分配,我们同样录制以上的一段操作。

详解Vue依赖收集引发的问题

此时我们能够清楚的看到,的确是这个Observer在作怪。同时,我们可以看到这是vue的代码,点击右边的文件查看source code,就可以清楚的明白这就是vue在执行依赖收集的操作,此时会给属性添加watcher。那我们这里为什么会有如此多的属性被添加了watcher呢?看了一下代码,原来是我把go.js的一个实例挂到了vue的data选项中,放到data中的属性会被vue执行依赖收集的相关操作,而这个实例拥有非常多的嵌套属性,全部都会被添加watcher。其实,我们只是想单纯的存储一下这个实例,供我们后续调用其相关的方法,添加watcher对我们来说完全没有意义,那我们如何避免这样的问题呢?

解决问题

上网搜索了相关的解决方案,大概有如下几种:

  1. 在data中定义的属性前面加上$,即通知vue该属性不需要被依赖收集,例如:javascript data() { return { $goDiagram: null } }但是这样声明,在template中引用时会报找不到$goDiagram属性的错误,具体的原因我还没深究,有空可以研究一下。
  2. 不在data中声明,直接在赋值的时候声明this.goDiagram = diagram。这同样会遇到第一种方案的问题,模板中会提示找不到goDiagram属性。
  3. 不在data中声明,而是利用$options来存储goDiagram,例如:
export default {
 goDiagram: null,
 mounted() {
  this.$options.goDiagram = xxx
 }
}

 这应该是比较好的一个方法,vue官方中也说明了$options用来包含自定义属性,例如我们平时引入的常量或是枚举类型,我们也不希望它们被添加无意义的watcher,因此可以通过这种方式来定义,在template中引用时只需要{{$options.xxx}}即可。这种方式唯一的缺点就是不能像data那样一眼望去就能清楚地知道你定义了什么属性。
项目中我采用了第一种方式,经过修改后内存占用量减少到原来的1/5到1/6,可以说效果非常好,再也不会出现浏览器崩溃的情况了。

总结

通过这样的一个问题,我们主要能够学习到两点:

  1. 如何通过chrome的开发者工具,去快速地定位代码中存在的内存问题
  2. 不要盲目的将属性都挂载到data选项中,一些常量我们可以采取上面提到的几种方式来定义,以此来作为一种优化手段

以上所述是小编给大家介绍的Vue依赖收集引发的问题详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
js 无提示关闭浏览器页面的代码
Mar 09 Javascript
表单切换,用回车键替换Tab健(不支持IE)
Jul 20 Javascript
js setTimeout opener的用法示例详解
Oct 23 Javascript
jquery对象和javascript对象即DOM对象相互转换
Aug 07 Javascript
Javascript显示和隐藏ul列表的方法
Jul 15 Javascript
JavaScript实现为input与textarea自定义hover,focus效果的方法
Aug 21 Javascript
IE8下jQuery改变png图片透明度时出现的黑边
Aug 30 Javascript
基于JS代码实现当鼠标悬停表格上显示这一格的全部内容
Jun 12 Javascript
Bootstrap标签页(Tab)插件切换echarts不显示问题的解决
Jul 13 Javascript
vue轻量级框架无法获取到vue对象解决方法
May 12 Javascript
简单了解JavaScript弹窗实现代码
May 07 Javascript
JavaScript中reduce()的用法
May 11 Javascript
JS大坑之19位数的Number型精度丢失问题详解
Apr 22 #Javascript
Vue $mount实战之实现消息弹窗组件
Apr 22 #Javascript
深入理解vue中的slot与slot-scope
Apr 22 #Javascript
浅析vue插槽和作用域插槽的理解
Apr 22 #Javascript
详解50行代码,Node爬虫练手项目
Apr 22 #Javascript
Vue匿名插槽与作用域插槽的合并和覆盖行为
Apr 22 #Javascript
详解Vue 匿名、具名和作用域插槽的使用方法
Apr 22 #Javascript
You might like
php 冒泡排序 交换排序法
2011/05/10 PHP
简单谈谈php中ob_flush和flush的区别
2014/11/27 PHP
php导入excel文件到mysql数据库的方法
2015/01/14 PHP
php绘图之在图片上写中文和英文的方法
2015/01/24 PHP
PHP响应post请求上传文件的方法
2015/12/17 PHP
PHP实现的oracle分页函数实例
2016/01/25 PHP
CI框架整合smarty步骤详解
2016/05/19 PHP
php关联数组与索引数组及其显示方法
2018/03/12 PHP
js showModalDialog参数的使用详解
2014/01/07 Javascript
JS实现跟随鼠标立体翻转图片的方法
2015/05/04 Javascript
异步JavaScript编程中的Promise使用方法
2015/07/28 Javascript
jQuery实现侧浮窗与中浮窗切换效果的方法
2016/09/05 Javascript
浅谈关于axios和session的一些事
2017/07/13 Javascript
关于Vue Router中路由守卫的应用及在全局导航守卫中检查元字段的方法
2018/12/09 Javascript
原生js实现贪食蛇小游戏的思路详解
2019/11/26 Javascript
vue中全局路由守卫中替代this操作(this.$store/this.$vux)
2020/07/24 Javascript
vue点击按钮实现简单页面的切换
2020/09/08 Javascript
在实例中重学JavaScript事件循环
2020/12/03 Javascript
js闭包的9个使用场景
2020/12/29 Javascript
对Python中type打开文件的方式介绍
2018/04/28 Python
Pycharm设置utf-8自动显示方法
2019/01/17 Python
PyCharm GUI界面开发和exe文件生成的实现
2020/03/04 Python
Python识别处理照片中的条形码
2020/11/16 Python
浅析python连接数据库的重要事项
2021/02/22 Python
Adobe Html5 Extension开发初体验图文教程
2017/11/14 HTML / CSS
澳大利亚最好的厨具店:Kitchen Warehouse
2018/03/13 全球购物
LUISAVIAROMA中国官网:时尚奢侈品牌购物网站
2020/11/01 全球购物
医药销售求职信范文
2014/02/01 职场文书
数控技术学生的自我评价
2014/02/15 职场文书
学期评语大全
2014/04/30 职场文书
保护环境建议书100字
2014/05/13 职场文书
中央空调节能方案
2014/06/15 职场文书
关于青春的演讲稿800字
2014/08/22 职场文书
2015年公务员转正工作总结
2015/04/24 职场文书
Python激活Anaconda环境变量的详细步骤
2021/06/08 Python
MySQL学习之基础操作总结
2022/03/19 MySQL