當我們給全域變數賦值null,從而主動解除引用,釋放內存,那麼這屬於引用計數策略嗎?不是說JS的垃圾回收機制只用標記清除嗎?標記清除策略裡就不涉及解除引用嗎?
變數宣告會在堆上分配內存,將相應引用標記為 null 是可以告訴解釋器,這個變數對應的內存空間是可以被回收的。
不過這是 JS 的語法設計,並不涉及解釋器實作中記憶體管理的細節。實際上,瀏覽器的 GC 策略也是有差異的。 Chrome / Firefox / Safari 採用標記清除,而舊版 IE 採用參考計數。
並且,解除引用也不僅僅只有賦值 null 的形式。解釋器可以透過作用域的方式決定變數的生命週期,並在離開變數作用域時回收變數的記憶體空間。
【標記清除】和【引用計數】是兩種不同的 GC 演算法,而【解除引用】則是 JS 中的語法特性,二者可以是正交(無關)的。
js規範是用標記清除,但實作的話不一定都是標記清除。
而你這裡疑惑的原因是因為只看到回收表象,沒看清除本質。
引用計數顧名思義就是對物件的引用進行計數,當引用為0時則回收。
而標記清除則分兩個階段,標記階段從根開始遍歷,對於可以訪問到的對像比較為可達對象,然後在清除階段將那些沒有被標記的對象回收。
其實你除非去看原始碼實現,不然很難從表像看出用的是什麼策略。
變數宣告會在堆上分配內存,將相應引用標記為 null 是可以告訴解釋器,這個變數對應的內存空間是可以被回收的。
不過這是 JS 的語法設計,並不涉及解釋器實作中記憶體管理的細節。實際上,瀏覽器的 GC 策略也是有差異的。 Chrome / Firefox / Safari 採用標記清除,而舊版 IE 採用參考計數。
並且,解除引用也不僅僅只有賦值 null 的形式。解釋器可以透過作用域的方式決定變數的生命週期,並在離開變數作用域時回收變數的記憶體空間。
【標記清除】和【引用計數】是兩種不同的 GC 演算法,而【解除引用】則是 JS 中的語法特性,二者可以是正交(無關)的。
js規範是用標記清除,但實作的話不一定都是標記清除。
而你這裡疑惑的原因是因為只看到回收表象,沒看清除本質。
引用計數顧名思義就是對物件的引用進行計數,當引用為0時則回收。
而標記清除則分兩個階段,標記階段從根開始遍歷,對於可以訪問到的對像比較為可達對象,然後在清除階段將那些沒有被標記的對象回收。
其實你除非去看原始碼實現,不然很難從表像看出用的是什麼策略。