Welcome to OStack Knowledge Sharing Community for programmer and developer-Open, Learning and Share
Welcome To Ask or Share your Answers For Others

Categories

0 votes
1.9k views
in Technique[技术] by (71.8m points)

js 求解一个闭包内存泄露问题

function test() {
    var a = [];
    return function (){
        a.push(1)
        return a
    }
}
var t = test();
t();
t();
t = null;

这样的话 a 会被释放吗? 如何不会应该怎么释放?


与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
Welcome To Ask or Share your Answers For Others

1 Answer

0 votes
by (71.8m points)

这是写 t=null时的图

image.png

这是不写 t = null时的图

image.png
以上都是在无痕模式下6秒的结果
可以发现无论是否写t = null, 都是没发生内存泄漏的。

多次测试, 偶尔有以下结果
image.png

可以发现,有创建堆栈,但是是趋于稳定的,是绝对没发生内存泄漏的。
也许这就是v8的威力吧

以上测试是在遍历1w遍的情况下产生的。
再次根据次数测试

测试

1W

for (let i = 0; i < 10000; i++) {
    t();
  }

image.png
最终被释放, 且最高时占用1.6mb

100w

for (let i = 0; i < 1000000; i++) {
    t();
  }

image.png
最终被释放,且最高时占用2.8mb,最终heap为883kb

10000w

for (let i = 0; i < 100000000; i++) {
    t();
  }

image.png
最终被释放,且最高时占用还是2.8mb,最终heap为883kb

修改代码测试

将t的引用放入循环会怎样?

 for (let i = 0; i < 100000000; i++) {
    var t = test();
    t();
  }

image.png
可以看到确实比把引用写在外面多占用了很多内存,但最终还是平稳趋于2.0mb,所以并未造成内存飞升,只是t的引用没有释放.

手动将t引用释放!

for (let i = 0; i < 100000000; i++) {
    var t = test();
    t();
    t = null
  }

image.png

芜湖~ 起飞 heap又回到了883kb ~


与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
Welcome to OStack Knowledge Sharing Community for programmer and developer-Open, Learning and Share
Click Here to Ask a Question

2.1m questions

2.1m answers

60 comments

57.0k users

...