最近看了一些 JavaScript 的内存泄露问题,看似没问题的代码原来存在内存泄露,而且部分还不知道怎么回事,比如:
<span class="kd">function</span> <span class="p">(</span><span class="nx">element</span><span class="p">,</span><span class="nx">a</span><span class="p">,</span><span class="nx">b</span><span class="p">){</span> <span class="nx">element</span><span class="p">.</span><span class="nx">onclick</span> <span class="o">=</span> <span class="kd">function</span><span class="p">(){</span> <span class="c1">//TODO a b here</span> <span class="p">}</span> <span class="p">}</span>
回复内容:
这个不叫「内存泄漏」。这个代码运行之后,只要 element 不再被引用,a、b 也会被回收。题主的意图估计是希望 a、b 的生命周期比 element 短。那是你的设计错误。因为你把 element 的一个 event-handler 设计成依赖于 a、b,那 a、b 当然就要和 element 共生死了。题主给的这个逻辑用不用闭包都会有这个问题。如果硬要释放 a、b,那就是 release before use,会造成 null-dereference error。 很多资料都过时了,内存泄露这个点在js圈子里更多的是惯性而不是实际影响。一般情况下,不用担心那么多,浏览器有自己的垃圾回收机制,只要你不作死把所有不再需要的资源都放到某个永远不会释放的数组什么的里面就好。遇到具体的内存泄露问题再具体解决。
<span class="kd">var</span> <span class="nx">test_obj</span> <span class="o">=</span> <span class="p">{</span>
<span class="nx">closure_fn</span><span class="o">:</span> <span class="kd">function</span> <span class="p">()</span> <span class="p">{</span>
<span class="kd">var</span> <span class="nx">that</span> <span class="o">=</span> <span class="k">this</span><span class="p">;</span>
<span class="kd">var</span> <span class="nx">val</span> <span class="o">=</span> <span class="nx">setTimeout</span><span class="p">(</span><span class="kd">function</span> <span class="p">()</span> <span class="p">{</span>
<span class="nx">console</span><span class="p">.</span><span class="nx">log</span><span class="p">(</span><span class="s1">'Rambo!'</span><span class="p">);</span>
<span class="nx">that</span><span class="p">.</span><span class="nx">closure_fn</span><span class="p">();</span>
<span class="p">},</span> <span class="mi">1000</span><span class="p">);</span>
<span class="p">}</span>
<span class="p">};</span>
<span class="nx">test_obj</span><span class="p">.</span><span class="nx">closure_fn</span><span class="p">();</span>
<span class="nx">test_obj</span> <span class="o">=</span> <span class="kc">null</span><span class="p">;</span>
<span class="c1">// 尝试之后,你会发现这他妈才是可怕的。</span>
<span class="kd">var</span> <span class="nx">test_obj_2</span> <span class="o">=</span> <span class="p">{</span>
<span class="nx">closure_fn</span><span class="o">:</span> <span class="kd">function</span> <span class="p">()</span> <span class="p">{</span>
<span class="kd">var</span> <span class="nx">that</span> <span class="o">=</span> <span class="nx">test_obj_2</span><span class="p">;</span>
<span class="kd">var</span> <span class="nx">val</span> <span class="o">=</span> <span class="nx">setTimeout</span><span class="p">(</span><span class="kd">function</span> <span class="p">()</span> <span class="p">{</span>
<span class="nx">console</span><span class="p">.</span><span class="nx">log</span><span class="p">(</span><span class="s1">'Rambo!'</span><span class="p">);</span>
<span class="nx">that</span> <span class="o">?</span> <span class="nx">that</span><span class="p">.</span><span class="nx">closure_fn</span><span class="p">()</span> <span class="o">:</span> <span class="nx">clearTimeout</span><span class="p">(</span><span class="nx">val</span><span class="p">);</span>
<span class="p">},</span> <span class="mi">1000</span><span class="p">);</span>
<span class="p">}</span>
<span class="p">};</span>
我记得 @贺师俊 老师在ITEYE上有过一段关于这个的讨论。首先,这是一个闭包。
题主贴的代码里之所以造成内存泄露是因为循环引用。一个闭包在创建时会附有三个属性:VO、thisValue、scope chain,而其中scope chain是指向外层parent scope的引用。
因此这个闭包实际上保存了外层函数中element的引用,而element本身又引用了闭包,循环引用因此而见。
但是我记得hax同时提到,因为循环引用导致的泄露实际上是IE浏览器引擎的bug而导致的,而如果element不是dom对象,不产生循环引用也就不会leak了。 这个不好说,有可能会泄露吧。。 我也有话说:
首先,IE9之前的版本因为对JScript对象和COM对象使用不同的垃圾收集机制,因此才有机会出现这个问题。再者,闭包的作用域链中保存着html元素,且包含循环引用。第三,取消循环引用并置空引用的html元素就可有效避免 仅供参考,未作过实验验证(附近确实找不到装IE6和IE7的机器了):
据微软称这是IE6在XP上特有的问题,照下面这篇最后更新日期为2011年10月的kb文章
A memory leak occurs in Internet Explorer 6 when you view a Web page that uses JScript scripting on a Windows XP-based computer
的说法,如果你的系统包含了这个更新:
http://support.microsoft.com/kb/947864
就没有这个问题了。 其实我很好奇的是,闭包,在什么情况下才会真的造成内存泄漏呢?
或许是我代码写少了,真的重来没有过内存泄漏的时候... 首先,题主问的闭包是否都会内存泄露,这个可以明确说不是。题主贴的代码产生了内存泄露是因为循环引用。低版本ie对这种循环引用的处理有bug,会造成内存泄露。
想要避免泄露,应该break circle。让element = null。这样handler就不持有dom的引用了。 内存泄漏和内存使用这两个概念搞清楚?

Untuk memaksimumkan kecekapan pembelajaran Python dalam masa yang terhad, anda boleh menggunakan modul, masa, dan modul Python. 1. Modul DateTime digunakan untuk merakam dan merancang masa pembelajaran. 2. Modul Masa membantu menetapkan kajian dan masa rehat. 3. Modul Jadual secara automatik mengatur tugas pembelajaran mingguan.

Python cemerlang dalam permainan dan pembangunan GUI. 1) Pembangunan permainan menggunakan pygame, menyediakan lukisan, audio dan fungsi lain, yang sesuai untuk membuat permainan 2D. 2) Pembangunan GUI boleh memilih tkinter atau pyqt. TKInter adalah mudah dan mudah digunakan, PYQT mempunyai fungsi yang kaya dan sesuai untuk pembangunan profesional.

Python sesuai untuk sains data, pembangunan web dan tugas automasi, manakala C sesuai untuk pengaturcaraan sistem, pembangunan permainan dan sistem tertanam. Python terkenal dengan kesederhanaan dan ekosistem yang kuat, manakala C dikenali dengan keupayaan kawalan dan keupayaan kawalan yang mendasari.

Anda boleh mempelajari konsep pengaturcaraan asas dan kemahiran Python dalam masa 2 jam. 1. Belajar Pembolehubah dan Jenis Data, 2.

Python digunakan secara meluas dalam bidang pembangunan web, sains data, pembelajaran mesin, automasi dan skrip. 1) Dalam pembangunan web, kerangka Django dan Flask memudahkan proses pembangunan. 2) Dalam bidang sains data dan pembelajaran mesin, numpy, panda, scikit-learn dan perpustakaan tensorflow memberikan sokongan yang kuat. 3) Dari segi automasi dan skrip, Python sesuai untuk tugas -tugas seperti ujian automatik dan pengurusan sistem.

Anda boleh mempelajari asas -asas Python dalam masa dua jam. 1. Belajar pembolehubah dan jenis data, 2. Struktur kawalan induk seperti jika pernyataan dan gelung, 3 memahami definisi dan penggunaan fungsi. Ini akan membantu anda mula menulis program python mudah.

Bagaimana Mengajar Asas Pengaturcaraan Pemula Komputer Dalam masa 10 jam? Sekiranya anda hanya mempunyai 10 jam untuk mengajar pemula komputer beberapa pengetahuan pengaturcaraan, apa yang akan anda pilih untuk mengajar ...

Cara mengelakkan dikesan semasa menggunakan fiddlerevery di mana untuk bacaan lelaki-dalam-pertengahan apabila anda menggunakan fiddlerevery di mana ...


Alat AI Hot

Undresser.AI Undress
Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover
Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Undress AI Tool
Gambar buka pakaian secara percuma

Clothoff.io
Penyingkiran pakaian AI

AI Hentai Generator
Menjana ai hentai secara percuma.

Artikel Panas

Alat panas

Muat turun versi mac editor Atom
Editor sumber terbuka yang paling popular

ZendStudio 13.5.1 Mac
Persekitaran pembangunan bersepadu PHP yang berkuasa

Pelayar Peperiksaan Selamat
Pelayar Peperiksaan Selamat ialah persekitaran pelayar selamat untuk mengambil peperiksaan dalam talian dengan selamat. Perisian ini menukar mana-mana komputer menjadi stesen kerja yang selamat. Ia mengawal akses kepada mana-mana utiliti dan menghalang pelajar daripada menggunakan sumber yang tidak dibenarkan.

EditPlus versi Cina retak
Saiz kecil, penyerlahan sintaks, tidak menyokong fungsi gesaan kod

Dreamweaver CS6
Alat pembangunan web visual