详解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 相关文章推荐
javascript cookies操作集合
Apr 12 Javascript
JavaScript 原型链学习总结
Oct 29 Javascript
js实现Select列表内容自动滚动效果代码
Aug 20 Javascript
使用jQuery的easydrag插件实现可拖动的DIV弹出框
Feb 19 Javascript
js实现类bootstrap模态框动画
Feb 07 Javascript
JS获取一个表单字段中多条数据并转化为json格式
Oct 17 Javascript
vue axios数据请求及vue中使用axios的方法
Sep 10 Javascript
vue双向绑定及观察者模式详解
Mar 19 Javascript
WebGL学习教程之Three.js学习笔记(第一篇)
Apr 25 Javascript
JavaScript解析JSON数据示例
Jul 16 Javascript
一行JavaScript代码如何实现瀑布流布局
Dec 11 Javascript
详解Javascript实践中的命令模式
May 05 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
ezSQL PHP数据库操作类库
2010/05/16 PHP
php根据指定位置和长度获得子字符串的方法
2015/03/17 PHP
学习php设计模式 php实现门面模式(Facade)
2015/12/07 PHP
[原创]PHP简单开启curl的方法(测试可行)
2016/01/11 PHP
PHP多进程编程实例详解
2017/07/19 PHP
通过Jquery遍历Json的两种数据结构的实现代码
2011/01/19 Javascript
jQuery 网易相册鼠标移动显示隐藏效果实现代码
2013/03/31 Javascript
JavaScript获取FCK编辑器信息的具体方法
2013/07/12 Javascript
Javascript基础 函数“重载” 详细介绍
2013/10/25 Javascript
Javascript 按位取反运算符 (~)
2014/02/04 Javascript
深入分析原生JavaScript事件
2014/12/29 Javascript
老生常谈JavaScript数组的用法
2016/06/10 Javascript
jQuery实现遍历复选框的方法示例
2017/03/06 Javascript
jQuery extend()详解及简单实例
2017/05/06 jQuery
浅谈对Angular中的生命周期钩子的理解
2017/07/31 Javascript
详解微信小程序中的页面代码中的模板的封装
2017/10/12 Javascript
使用命令行工具npm新创建一个vue项目的方法
2017/12/27 Javascript
详解vue中axios的封装
2018/07/18 Javascript
AngularJS实现与后台服务器进行交互的示例讲解
2018/08/13 Javascript
详解JavaScript添加给定的标签选项
2018/09/17 Javascript
分享5个小技巧让你写出更好的 JavaScript 条件语句
2018/10/20 Javascript
Vue实现腾讯云点播视频上传功能的实现代码
2020/08/17 Javascript
动态创建类实例代码
2009/10/07 Python
python django框架中使用FastDFS分布式文件系统的安装方法
2019/06/10 Python
10个示例带你掌握python中的元组
2020/11/23 Python
HTML5+CSS3实现机器猫
2016/10/17 HTML / CSS
基于HTML5+Webkit实现树叶飘落动画
2017/12/28 HTML / CSS
德国婴儿推车和儿童安全座椅商店:BABYSHOP
2016/09/01 全球购物
英国最大的自有市场,比亚马逊便宜:Flubit
2019/03/19 全球购物
英国Flybe航空官网:欧洲最大的独立支线廉价航空公司
2019/07/15 全球购物
Alexandre Birman美国官网:亚历山大·伯曼
2019/10/30 全球购物
描述一下JVM加载class文件的原理机制
2013/12/08 面试题
初中英语教学反思
2014/01/25 职场文书
科技开发中心办公室主任岗位责任制
2014/02/10 职场文书
二年级班级文化建设方案
2014/05/10 职场文书
如何利用python实现Simhash算法
2022/06/28 Python