详解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 更新 JavaScript 数组的 uniq 方法
Jan 23 Javascript
ExtJS 2.0实用简明教程 之ExtJS版的Hello
Apr 29 Javascript
javascript中自定义对象的属性方法分享
Jul 12 Javascript
使用pjax实现无刷新更改页面url
Feb 05 Javascript
javascript中slice(),splice(),split(),substring(),substr()使用方法
Mar 13 Javascript
Javascript闭包(Closure)详解
May 05 Javascript
angularjs 表单密码验证自定义指令实现代码
Oct 27 Javascript
解决linux下node.js全局模块找不到的问题
May 15 Javascript
通过jquery toggleClass()属性制作文章段落更改背景颜色
May 21 jQuery
JavaScript继承与多继承实例分析
May 26 Javascript
layui表格分页 记录勾选的实例
Sep 02 Javascript
JS使用for in有序获取对象数据
May 19 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编写和读取XML的几种方式
2013/01/12 PHP
ThinkPHP分页类使用详解
2014/03/05 PHP
发一个自己用JS写的实用看图工具实现代码
2008/07/26 Javascript
JS版网站风格切换实例代码
2008/10/06 Javascript
Mootools 1.2教程 类(一)
2009/09/15 Javascript
根据出生日期自动取得星座的js代码
2010/07/20 Javascript
JS拖动技术 关于setCapture使用
2010/12/09 Javascript
jquery实现Ctrl+Enter提交表单的方法
2015/07/21 Javascript
js实现Select列表内容自动滚动效果代码
2015/08/20 Javascript
JavaScript添加随滚动条滚动窗体的方法
2016/02/23 Javascript
javascript实现数据双向绑定的三种方式小结
2017/03/09 Javascript
基于vue2.0+vuex的日期选择组件功能实现
2017/03/13 Javascript
微信小程序实现给循环列表添加点击样式实例
2017/04/26 Javascript
JavaScript箭头(arrow)函数详解
2017/06/04 Javascript
JS实现微信里判断页面是否被分享成功的方法
2017/06/06 Javascript
教你用Cordova打包Vue项目的方法
2017/10/17 Javascript
vue2.0在table中实现全选和反选的示例代码
2017/11/04 Javascript
webpack4与babel配合使es6代码可运行于低版本浏览器的方法
2018/10/12 Javascript
[01:37]全新的一集《真视界》——TI7总决赛
2017/09/21 DOTA
浅谈python配置与使用OpenCV踩的一些坑
2018/04/02 Python
django 发送邮件和缓存的实现代码
2018/07/18 Python
Python字符串逆序输出的实例讲解
2019/02/16 Python
Python-ElasticSearch搜索查询的讲解
2019/02/25 Python
python3对接mysql数据库实例详解
2019/04/30 Python
如何解决django-celery启动后迅速关闭
2019/10/16 Python
python如何删除文件、目录
2020/06/23 Python
Elasticsearch py客户端库安装及使用方法解析
2020/09/14 Python
Python绘制数码晶体管日期
2021/02/19 Python
用CSS3实现无限循环的无缝滚动的实例代码
2017/07/04 HTML / CSS
英国度假别墅预订:Sykes Cottages
2017/06/12 全球购物
制药工程专业职业生涯规划范文
2014/03/10 职场文书
企业标语大全
2014/07/01 职场文书
人大调研汇报材料
2014/08/14 职场文书
交通事故代理词范文
2015/05/23 职场文书
门球健将观后感
2015/06/16 职场文书
再次探讨go实现无限 buffer 的 channel方法
2021/06/13 Golang