javascript錯誤的認知不用關心記憶體管理_jquery
介紹
低層次的語言,如C,具有低階的記憶體管理指令,如:malloc()和free(),需要開發者手動釋放記憶體。然而像javascript這樣的高階語言情況則不同,物件(objects, strings 等)創建的時候分配內存,當他們不在使用的時候內存會被自動回收,這個自動回收的過程被稱為垃圾回收。因為垃圾回收的存在,讓javascript等高階語言開發者產生了一個錯誤的認識,以為可以不用關心記憶體管理。
記憶體生命週期
不管什麼樣的程式語言,記憶體的生命週期基本上是一致的。
分配你需要的記憶體
使用他進行讀寫操作
當記憶體不需要的時候,釋放資源
步驟1和步驟2對於所有語言都一樣,能明顯覺察到。至於步驟3,低階語言需要開發者明確執行。而對於像javascript這樣的高階語言,這部分操作是交給解析器完成的,所以你不會覺察到。
javascript中的分配操作
值的初始化
在變數賦值的時候,javascript會完成記憶體的分配工作。
var n = 123; // 為數字分配記憶體🎜>var s = "azerty"; // 為字串分配記憶體
var o = {
a: 1,
b: null
}; // 為包含屬性值的object物件分配記憶體
var a = [1, null, "abra"]; // 為包含值的陣列分配記憶體
function f(a){
return a 2;
} // 為函數分配記憶體(函數是可呼叫的物件)
// 函數表達式同樣也是對象,存在分配記憶體的情況
someElement.addEventListener('click', function(){
someElement.style. backgroundColor = 'blue';
}, false);
一些函數當執行完畢之後,同樣存在物件分配的情況發生。
var d = new Date();
var e = document.createElement('div'); // 指派一個DOM 元素
程式碼如下:
var s = "azerty";
var s = "azerty";
對值的使用
對值的使用,其實也就是對分配後的記憶體執行讀寫操作。這些操作包括:對變數或物件的屬性進行讀寫操作,或是向函數傳遞參數。
當不再需要的時候,釋放記憶體
絕大多數記憶體管理的問題都發生在這個階段。最難做的事情是,如何判定分配的記憶體不再需要。這往往需要開發者做出判定,程式在什麼時候不再需要內存,並釋放他所佔資源。
高階語言的解析器中嵌入了一個叫做「垃圾收集器」的程序,他的工作是用來追蹤記憶體的分配和使用,判定記憶體是否被需要,在不再需要的時候執行資源釋放操作。他只能得到一個近似值,因為判斷一個記憶體是否被需要,這是個不確定的問題(不能用一種演算法解決)。
垃圾回收
如上文所述,我們無法準確的做到自動判定「記憶體不再需要」。所以,垃圾回收對該問題的解決方案有其限制。本節將解釋必要的概念,以了解主要的垃圾收集演算法和它們的限制。
引用
垃圾回收中一個主要的概念是引用。在記憶體管理中,當一個物件無論是顯式的還是隱式的使用了另一個對象,我們就說他引用了另一個對象。例如,javascript物件存在一個隱式的指向原型的引用,還有明確指向他的屬性值的引用。
在這裡,物件的概念超越了javascript傳統意義上物件的概念,他還包括函數作用域和全域作用域。
使用引用計數演算法的垃圾回收
下面要介紹的是一種最理想化的演算法,引入了「物件不再需要」 和「沒有其他物件引用該對象” 的概念。當該物件的引用指標變成0的時候,就認為他可以被回收。
範例:
var o = { a: {
b:2
}
}; // 建立了兩個物件. 一個物件(a)被另一個物件(o引用的物件)引用,並把a當作他的屬性
// 該物件又被變數o引用
// 很明顯,這時沒有物件能被回收
var o2 = o; // 變數o2 再次引用了該物件
o = 1 ; // o 不再引用該對象,只有o2還在引用該對象
var oa = o2.a; // oa引用o2 的屬性對象a
// 該對像被其他兩個對象引用,分別是o2的屬性a和oa變數
o2 = "yo"; // 該物件已經不再被其他物件引用了,但是他的屬性a任然被oa變數引用,所以他還不能被釋放
oa = null; // 現在屬性a也不再被別的物件引用,該物件可以被回收了
限制:循環 該演算法有其局限性,當一個對象引用另一個對象,當形成循環引用時,即時他們不再被需要了,垃圾收集器也不會回收他們。
var o };
var o2 = {};
o.a = o2; // o 引用o2
o2.a = o; // o2 引用o
return "azerty";
}
f();
// 兩個物件被創建,並形成相互引用
// 函數呼叫結束之後,他們不會脫離函數作用域,雖然他們不會被使用,但不會被釋放
// 這是因為,引用計數的演算法判定只要物件存在被引用的情況,那麼就不能對其執行垃圾回收
ie6、7中,在dom物件上使用引用計數的演算法,這裡會存在記憶體外洩的問題。
div.onclick = function(){
doSomething();
}; // div 透過click 屬性引用了事件處理程序
// 當事件處理函數中存取了div變數的時候,會形成循環引用,將導致兩個物件都不會被回收,造成記憶體外洩
標記 - 清除演算法
他引入了「物件不再需要」和「物件無法存取(物件不可達)」的概念。演算法假設有一系列的根對象(javascript中的根對象就是全域對象),每隔一段時間,垃圾收集器就會從根對象開始,遍歷所以他引用的對象,然後再遍歷引用對象引用的對象,以此類推。使用這種方式,垃圾收集器可以獲得所有可存取的對象,回收那些無法存取的對象。
這種演算法比之前的演算法好些,0引用的對象會被設定為不可訪問對象,同時他也避免了循環引用造成的困惱。
截止2012年,大多數現代瀏覽器使用的是這種「標記-清除演算法」的垃圾回收器。 JavaScript垃圾收集領域(代/增量/並發/並行的垃圾收集),在過去的幾年改善了與之相關的演算法,但是垃圾收集演算法本身(標記-清除演算法)和「如何判定一個物件不再需要」並沒有得以改善。
週期不再是一個問題
在第一個例子中,函數呼叫結束之後,這兩個物件不會被全域物件引用,也不會被全域對象引用的物件引用。因此,他們會被javascript垃圾回收器標記為不可存取物件。這種事情同樣也發生在第二個例子中,當div和事件處理函數被垃圾回收器標記為不可訪問,他們就會被釋放掉。
限制:物件需要明確的標記為不可訪問
這種標記的方法存在局限,但是我們在程式設計中被沒有接觸到他,所以我們很少關心垃圾回收相關的內容。

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

C++物件佈局和記憶體對齊優化記憶體使用效率:物件佈局:資料成員按聲明順序存儲,優化空間利用率。記憶體對齊:資料在記憶體中對齊,提升存取速度。 alignas關鍵字指定自訂對齊,例如64位元組對齊的CacheLine結構,提高快取行存取效率。

C++函數記憶體分配和銷毀的最佳實踐包括:使用局部變數進行靜態記憶體分配。使用智慧指標進行動態記憶體分配。在建構函式中分配內存,在析構函式中銷毀記憶體。使用自訂記憶體管理器進行複雜記憶體場景。使用異常處理進行資源清理,確保在異常時釋放已分配記憶體。

C++中的自訂記憶體分配器可讓開發者根據需求調整記憶體分配行為,建立自訂分配器需要繼承std::allocator並重寫allocate()和deallocate()函式。實戰案例包括:提高效能、優化記憶體使用和實現特定行為。使用時需要注意避免釋放內存,管理內存對齊,並進行基準測試。

引用計數機制在C++記憶體管理中用於追蹤物件的引用情況並自動釋放未使用記憶體。此技術為每個物件維護一個引用計數器,當引用新增或移除時計數器會相應增減。當計數器降為0時,物件被釋放,無需手動管理。但循環引用會導致記憶體洩漏,維護引用計數器會增加開銷。

在多執行緒環境中,C++記憶體管理面臨以下挑戰:資料競爭、死鎖和記憶體洩漏。因應措施包括:1.使用同步機制,如互斥鎖和原子變數;2.使用無鎖資料結構;3.使用智慧指標;4.(可選)實現垃圾回收。

C++函數記憶體管理提供了擴充和進階技術,包括:自訂分配器:允許使用者定義自己的記憶體分配策略。 placementnew和placementdelete:當需要將物件分配到特定記憶體位置時使用。進階技術:記憶體池、智慧指標和RAII,用於減少記憶體洩漏、提高效能和簡化程式碼。

PHP函數中管理記憶體佔用需:避免宣告不必要的變數;使用輕量級資料結構;釋放未使用的變數;最佳化字串處理;限制函數參數;最佳化循環和條件,例如避免死循環和使用索引數組。

Go中函數的記憶體以值傳遞,不會影響原始變數。 Goroutine共享內存,其分配的內存不會被GC回收,直到Goroutine完成執行。記憶體洩漏可能發生在持有已完成的Goroutine引用、使用全域變數或避免靜態變數的情況下。為了避免洩漏,建議透過通道取消Goroutine、避免靜態變數、使用defer語句來釋放資源。
