深入了解JavaScript中的垃圾回收機(jī)制
JavaScript中的垃圾回收機(jī)制負(fù)責(zé)自動(dòng)管理內(nèi)存,回收不再使用的對(duì)象所占用的內(nèi)存空間。在JavaScript中,開(kāi)發(fā)者不需要顯式地分配和釋放內(nèi)存,垃圾回收器會(huì)自動(dòng)完成這些操作。以下是關(guān)于JavaScript垃圾回收機(jī)制的一些關(guān)鍵概念:
- 內(nèi)存生命周期:JavaScript內(nèi)存生命周期包括分配、使用和釋放三個(gè)階段。首先,內(nèi)存會(huì)被分配給變量或?qū)ο?;然后,程序?huì)使用這些變量或?qū)ο?;最后,不再需要的變量或?qū)ο髸?huì)被垃圾回收器釋放。
- 可達(dá)性:垃圾回收器通過(guò)可達(dá)性來(lái)判斷一個(gè)對(duì)象是否還在使用。根對(duì)象(如全局對(duì)象和其他內(nèi)置對(duì)象)被認(rèn)為是可達(dá)的。如果一個(gè)對(duì)象可以通過(guò)根對(duì)象或其他可達(dá)對(duì)象引用鏈到達(dá),那么它也被認(rèn)為是可達(dá)的。
- 引用計(jì)數(shù):這是一種較早的垃圾回收策略,通過(guò)追蹤每個(gè)對(duì)象的引用次數(shù)來(lái)判斷對(duì)象是否仍在使用。當(dāng)對(duì)象的引用計(jì)數(shù)為0時(shí),表示對(duì)象不再被使用,可以被回收。然而,引用計(jì)數(shù)算法存在循環(huán)引用問(wèn)題,無(wú)法回收循環(huán)引用的對(duì)象。
- 標(biāo)記-清除:這是現(xiàn)代JavaScript引擎中常見(jiàn)的垃圾回收算法。標(biāo)記-清除算法首先會(huì)標(biāo)記所有可達(dá)對(duì)象,然后遍歷整個(gè)內(nèi)存空間,清除未被標(biāo)記的對(duì)象。這種算法可以處理循環(huán)引用問(wèn)題,但可能會(huì)導(dǎo)致內(nèi)存碎片。
- 分代回收:由于不同對(duì)象的生命周期長(zhǎng)短不同,現(xiàn)代JavaScript引擎將內(nèi)存分為新生代和老生代。新生代主要存放短生命周期的對(duì)象,老生代主要存放長(zhǎng)生命周期的對(duì)象。新生代和老生代的垃圾回收策略會(huì)有所不同。
- 增量回收和懶惰回收:為了降低垃圾回收對(duì)程序執(zhí)行的影響,現(xiàn)代JavaScript引擎采用了增量回收和懶惰回收策略。增量回收將回收工作分成多個(gè)小任務(wù),穿插在程序執(zhí)行過(guò)程中;懶惰回收則會(huì)在一定程度上推遲回收操作,以減少性能開(kāi)銷。
以下是一個(gè)簡(jiǎn)單的示例,演示了 JavaScript 垃圾回收機(jī)制中的引用計(jì)數(shù)和標(biāo)記清除:
// 引用計(jì)數(shù)示例 let a = { name: 'John' }; let b = a; // b 引用了 a,a 的引用計(jì)數(shù)變?yōu)?2 a = null; // a 不再引用這個(gè)對(duì)象,a 的引用計(jì)數(shù)變?yōu)?1 b = null; // b 不再引用這個(gè)對(duì)象,這個(gè)對(duì)象的引用計(jì)數(shù)變?yōu)?0,可以被垃圾回收器回收 // 標(biāo)記清除示例 function foo() { let x = { name: 'Alice' }; let y = { name: 'Bob' }; x.friend = y; y.friend = x; } foo(); // 函數(shù)執(zhí)行完后,x 和 y 都不再被使用,但它們之間相互引用,無(wú)法使用引用計(jì)數(shù)來(lái)回收內(nèi)存 // 垃圾回收器定期運(yùn)行,會(huì)發(fā)現(xiàn) x 和 y 都已經(jīng)不再被引用,可以被回收
在這個(gè)示例中,當(dāng)變量 a
被賦值給變量 b
時(shí),對(duì)象的引用計(jì)數(shù)變?yōu)?2。當(dāng) a
被賦值為 null
時(shí),對(duì)象的引用計(jì)數(shù)變?yōu)?1。最后當(dāng) b
也被賦值為 null
時(shí),對(duì)象的引用計(jì)數(shù)變?yōu)?0,可以被垃圾回收器回收。
另外,函數(shù) foo
中創(chuàng)建了兩個(gè)對(duì)象 x
和 y
,并且它們相互引用。在函數(shù)執(zhí)行完后,這兩個(gè)對(duì)象不再被使用,但它們之間的引用關(guān)系無(wú)法使用引用計(jì)數(shù)來(lái)回收內(nèi)存。因此,垃圾回收器會(huì)定期運(yùn)行,查找那些已經(jīng)不再被引用的對(duì)象,然后釋放它們所占用的內(nèi)存空間。
再來(lái)一個(gè)例子,我們將創(chuàng)建一些對(duì)象并解釋JavaScript的垃圾回收機(jī)制。
// 創(chuàng)建對(duì)象 function createPerson(name, age) { return { name: name, age: age, }; } // 創(chuàng)建兩個(gè)對(duì)象 let person1 = createPerson("Alice", 30); let person2 = createPerson("Bob", 35); // person1 和 person2 變量引用了兩個(gè)新創(chuàng)建的對(duì)象,這些對(duì)象在內(nèi)存中是可達(dá)的 // 現(xiàn)在將 person1 引用另一個(gè)對(duì)象 person1 = createPerson("Charlie", 28); // 之前 person1 引用的 "Alice" 對(duì)象現(xiàn)在已經(jīng)不再可達(dá),因?yàn)闆](méi)有變量引用它 // JavaScript的垃圾回收器會(huì)識(shí)別到這一點(diǎn),并在合適的時(shí)機(jī)釋放其內(nèi)存 // 創(chuàng)建一個(gè)循環(huán)引用 let objA = { name: "ObjA", }; let objB = { name: "ObjB", }; objA.link = objB; objB.link = objA; // 將變量設(shè)置為 null,打破可達(dá)性 objA = null; objB = null; // 現(xiàn)在 objA 和 objB 對(duì)象都不再可達(dá),即使它們彼此引用 // 使用標(biāo)記-清除算法的垃圾回收器會(huì)識(shí)別到這一點(diǎn),并釋放它們占用的內(nèi)存
在這個(gè)例子中,我們創(chuàng)建了幾個(gè)對(duì)象并對(duì)它們進(jìn)行了引用。當(dāng)一個(gè)對(duì)象不再可達(dá)時(shí),它就成為了垃圾回收的目標(biāo)。對(duì)于循環(huán)引用的情況,標(biāo)記-清除算法可以識(shí)別到并正確處理這種情況,釋放不再使用的對(duì)象所占用的內(nèi)存。
注意:不同的JavaScript引擎可能采用不同的垃圾回收策略,如V8、SpiderMonkey和JavaScriptCore等。
到此這篇關(guān)于深入了解JavaScript中的垃圾回收機(jī)制的文章就介紹到這了,更多相關(guān)JavaScript垃圾回收機(jī)制內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
微信小程序?qū)崿F(xiàn)自定義picker選擇器彈窗內(nèi)容
這篇文章主要為大家詳細(xì)介紹了微信小程序?qū)崿F(xiàn)自定義picker選擇器彈窗內(nèi)容,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07js點(diǎn)擊列表文字對(duì)應(yīng)該行顯示背景顏色的實(shí)現(xiàn)代碼
這篇文章主要介紹了js點(diǎn)擊列表文字對(duì)應(yīng)該行顯示背景顏色的實(shí)現(xiàn)代碼,感興趣的小伙伴可以參考下2015-08-08JavaScript表格隔行變色和Tab標(biāo)簽頁(yè)特效示例【附j(luò)Query版】
這篇文章主要介紹了JavaScript表格隔行變色和Tab標(biāo)簽頁(yè)特效,結(jié)合實(shí)例形式分析了javascript元素遍歷、事件響應(yīng)相關(guān)操作技巧,并附帶jQuery對(duì)應(yīng)實(shí)現(xiàn)代碼供大家參考,需要的朋友可以參考下2019-07-07JavaScript實(shí)現(xiàn)移動(dòng)端橫豎屏檢測(cè)
這篇文章主要為大家詳細(xì)介紹了JavaScript實(shí)現(xiàn)移動(dòng)端橫豎屏檢測(cè),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-07-07location.href 在IE6中不跳轉(zhuǎn)的解決方法與推薦使用代碼
在js中,我們經(jīng)常使用location.href來(lái)實(shí)現(xiàn)頁(yè)面的跳轉(zhuǎn),為了方便我們寫(xiě)成函數(shù)。下面就分別說(shuō)明下,下面的一些代碼的實(shí)現(xiàn)問(wèn)題。2010-07-07layui當(dāng)點(diǎn)擊文本框時(shí)彈出選擇框,顯示選擇內(nèi)容的例子
今天小編就為大家分享一篇layui當(dāng)點(diǎn)擊文本框時(shí)彈出選擇框,顯示選擇內(nèi)容的例子,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-09-09利用Keydown事件阻止用戶輸入實(shí)現(xiàn)代碼
這篇文章主要介紹了利用Keydown事件阻止用戶輸入的具體實(shí)現(xiàn),需要的朋友可以參考下2014-03-03