最近发现项目有个bug,同时运行多个任务的时候,前端页面报内存不足而导致页面崩溃,这很明显就是内存泄露了。我查看了一下,运行的过程中,因为运行时间很久,所以前端和后台约定了,用计时器setInternal定时去请求后台运行状态,当运行状态为完成时,前端会清除定时器。我预估是因为计时器而导致的内存泄露,在执行计时器代码的时候,任务管理器的物理内存消耗一直在增加,这样的话,要是多个任务同时在执行,而且任务执行较久的话,那样物理内存就有可能会被占用完。后面我也复现了这个场景,果然是因为计时器的原因。
一、探究过程
疑惑1:会不会是因为JavaScript引擎是单线程,计时器会不断把事件不断放入事件队列,而任务执行时间很长,所以才会导致事件队列堆满而导致内存泄露?
资料答疑:首先,纠正大家一个错误的理解,定时器并不是严格意义上会按个多少秒执行的,它可能会出现执行延迟或提前。在运行本行代码的时候,将定时器的代码添加到了事件队列当中,而不是何时执行/运行代码。此时需要等到当前“事件处理程序”运行之后再去执行定时器代码。换句话说,就是,并非是设置的毫秒数后就执行定时器代码,执行的时间是有可能提前/延后的。同时,JavaScript引擎设置了:仅当队列中没有该定时器的任何其他代码实例时,才能够将定时器代码添加到队列。所以不会出现导致连续运行多次的情况。
疑惑2:那到底是什么导致内存泄露呢?
资料答疑:内联书写setInterval时,由于匿名函数被定义于全局中,不能够计时器的清除,因此很容易造成内存泄露。
二、实验测试
刚好我的计时器setInternal是用匿名函数写的,很有可能是因为这个原因,所以我用了匿名函数和命名函数测试了一下。
1、匿名函数
代码:
1 mounted(){
2 let self = this;
3 self.setInternalId = setInterval(()=>{
4 let sum = 0,i=1;
5 while(i<100000000){
6 sum+=i++;
7 // sum=parseInt(sum/2)
8 }
9 let now = new Date();
10 console.log(sum,'秒数:',now.getSeconds());
11 },2000);
12 },
13 beforeDestroy(){
14 let self = this
15 clearInterval(self.setInternalId);
16 console.log('消除定时器啦。。。。')
17 },
我发现物理内存是很缓慢增长的,所以要时间够长才能会有明显的区别,所以不能确定是匿名函数导致的。
2、命名函数
代码:
1 mounted(){
2 let self = this;
3 self.setInternalId = setInterval(this.getSum,2000);
4 },
5 beforeDestroy(){
6 let self = this
7 clearInterval(self.setInternalId);
8 console.log('消除定时器啦。。。。')
9 },
10 methods:{
11 getSum(){
12 let sum = 0,i=1;
13 while(i<100000000){
14 sum+=i++;
15 // sum=parseInt(sum/2)
16 }
17 let now = new Date();
18 console.log(sum,'秒数:',now.getSeconds());
19 }
20 }
我持续观察了半个多小时物理内存的变化,发现内存是时增长时减低,增长或降低的幅度都不会很大。
3、增加http请求
后面我在里面增加一个htttp请求后台数据,发现物理内存是一直上升的,上升的幅度明显比匿名函数大,上升的速度也很快,所以可能是前端请求导致的内存泄露。同时为了校验是setInternal还是http请求导致的内存泄露,我做了以下的代码校验:
1 mounted(){
2 let self = this;
3 self.setInternalId2 = setInterval(()=>{
4 let now = new Date();
5 console.log('第一个:',now.getSeconds());
6 self.getAllTasks(1); // http请求
7 self.setInternalId3 = setInterval(()=>{
8 let sum =0,i=1;
9 while(i<1000000){
10 sum+=i++;
11 }
12 let now = new Date();
13 console.log('第二个:',now.getSeconds());
14 },3000)
15 },1000)
16 },
17 beforeDestroy(){
18 let self = this;
19 clearInterval(self.setInternalId2);
20 clearInterval(self.setInternalId3);
21 console.log('消除计时器。。。。。')
22 },
发中间的setInternal计时器删掉后,物理内存消耗还是持续快速的增长。后面查资料得知,浏览器对单窗口的http请求数量是有限制的,谷歌浏览器可以并发执行最多6个,所以会导致http请求阻塞,从而消耗内存。
总结:这两个测试并不能证明是匿名函数所引起的内存泄露,很有可能是不断重复执行 ajax 引起的,但是我目前还查不到相关的资料证明,这个问题还有待考察。
三、常见的内存泄露
1、全局变量
在浏览器的环境下,全局变量对象就是window,定义全局变量如下:
1 function index(){
2 bar = "dsasd" ;
3 }
4 //上面代码相当于
5 function index(){
6 window.bar = "dsasd";
7 }
如果定义变量的时候忘记加上let或var,这时一个全局变量就会被创建出来,还有另一种定义全局变量
1 function foo(){
2 this.variable= "potential accidental global";
3 }
4 // 函数自身发生了调用,this 指向全局对象(window),(这时候会为全局对象 window 添加一个 variable 属性)而不是 undefined。
5
6 foo();
解决办法:为了防止这种错误的发生,在 JavaScript 文件开头添加 'use strict';
语句。这个语句实际上开启了解释 JavaScript 代码的严格模式,这种模式可以避免创建意外的全局变量。
2、定时器和回调函数
定时器造成内存泄露的主要原因是周期函数一直在运行,处理函数并不会被回收(只有周期函数停止运行之后才开始回收内存)。如果周期处理函数不能被回收,它的依赖程序也同样无法被回收。这意味着一些资源,也许是一些相当大的数据都也无法被回收。
观察者即监听事件,当它们不再被需要的时候(或者关联对象将要失效的时候)显式地将他们移除是十分重要的。现在,当观察者对象失效的时候便会被回收,即便 listener 没有被明确地移除,绝大多数的浏览器可以或者将会支持这个特性。尽管如此,在对象被销毁之前移除观察者依然是一个好的实践。
3、DOM 之外的引用
4、闭包