ホームページ ウェブフロントエンド jsチュートリアル JavaScriptのメモリ管理を詳しく分析_基礎知識

JavaScriptのメモリ管理を詳しく分析_基礎知識

May 16, 2016 pm 05:16 PM
javascript メモリ管理

はじめに

C などの低レベル言語には、malloc() や free() などの低レベルのメモリ管理コマンドがあり、開発者は手動でメモリを解放する必要があります。ただし、JavaScript などの高級言語では状況が異なります。オブジェクト (オブジェクト、文字列など) は作成時にメモリを割り当て、使用されなくなったメモリは自動的にリサイクルされます。この自動リサイクルプロセスはガベージコレクションと呼ばれます。 JavaScript などの高級言語の開発者は、ガベージ コレクションの存在により、メモリ管理を気にする必要がないと誤解しています。

メモリのライフサイクル

どのようなプログラミング言語であっても、メモリのライフサイクルは基本的に同じです。

1. 必要なメモリを割り当てます

2. 読み取りおよび書き込み操作に使用します

3. メモリが不要になったらリソースを解放します

ステップ 1 と 2 はすべての言語で同じであり、明確に気づくことができます。ステップ 3 に関しては、低レベル言語では開発者がこれを明示的に実行する必要があります。 JavaScript のような高水準言語の場合、操作のこの部分はパーサーによって実行されるため、気付かれることはありません。

JavaScript での代入操作

値の初期化

変数に値を代入すると、JavaScript によってメモリの割り当てが完了します。

コードをコピーします コードは次のとおりです。

var n = 123; // メモリを割り当てます。数値の場合
var s = "azerty"; // 文字列にメモリを割り当てます

var o = {
a: 1,
b: null
}; // 属性を含みますvalue オブジェクト 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 s2 = s.substr(0, 3); // s2 は新しい文字列です
// 文字列は変更されていないため、JavaScript は範囲 [0, 3]
var a = ["ouais ouais", "nan nan"];
var a2 = ["世代", "nan nan"];
var a3 = a.concat(a2) / / a と a2 を組み合わせて新しい配列を生成します


値の使用

値の使用は、実際には、割り当てられたメモリ上で読み取りおよび書き込み操作を実行することです。これらの操作には、変数またはオブジェクト プロパティに対する読み取りおよび書き込み操作、または関数へのパラメーターの受け渡しが含まれます。

不要になったらメモリを解放します

メモリ管理の問題のほとんどはこの段階で発生します。最も難しいのは、割り当てられたメモリがいつ不要になるかを判断する方法です。このため、開発者は多くの場合、プログラムがメモリを必要としなくなった時期を判断し、メモリが占​​有しているリソースを解放する必要があります。

「ガベージ コレクター」と呼ばれるプログラムは、高級言語パーサーに組み込まれており、メモリの割り当てと使用状況を追跡し、メモリが必要かどうかを判断し、メモリが不要になった場合はリソースの解放操作を実行します。必要です。メモリが必要かどうかの判断は不確実な問題である (アルゴリズムでは解決できない) ため、彼は近似値を取得することしかできません。

ガベージコレクション

上で述べたように、「メモリが不要になった」と正確かつ自動的に判断することはできません。したがって、この問題の解決策としてガベージ コレクションには限界があります。このセクションでは、主要なガベージ コレクション アルゴリズムとその制限事項を理解するために必要な概念について説明します。

引用

ガベージ コレクションの主要な概念は参照です。メモリ管理では、オブジェクトが明示的または暗黙的に別のオブジェクトを使用することを、そのオブジェクトが別のオブジェクトを参照すると言います。たとえば、JavaScript オブジェクトには、そのプロトタイプへの暗黙的な参照と、そのプロパティ値への明示的な参照があります。

ここでのオブジェクトの概念は、JavaScript における従来のオブジェクトの概念を超えており、関数スコープとグローバル スコープも含まれています。

参照カウント アルゴリズムを使用したガベージ コレクション

以下に紹介するのは、「オブジェクトが不要になった」、「他のオブジェクトがそのオブジェクトを参照しない」という概念を導入した最適なアルゴリズムです。オブジェクトの参照ポインタが 0 になると、リサイクルの準備ができたとみなされます。

例:

コードをコピー コードは次のとおりです。

var o = {
a: {
b:2
}
}; // 2 つのオブジェクトが作成され、1 つのオブジェクト (a) が別のオブジェクト (o で参照されるオブジェクト) によって参照されます。その属性
// オブジェクトは変数 o によって再び参照されます
// 明らかに、現時点ではオブジェクトをリサイクルできません


var o2 = o; // 変数 o2 は、再びオブジェクト
o = 1; // o はオブジェクトを参照しなくなり、o2 のみが引き続きオブジェクトを参照します

var oa = o2.a; // oa は o2 の属性オブジェクト a を参照します
// このオブジェクトは、他の 2 つのオブジェクト、つまり o2 の属性 a と oa 変数によって参照されています。

o2 = "yo"; // このオブジェクトは他のオブジェクトから参照されなくなりました。属性 a はまだ oa 変数によって参照されているため、まだ解放できません

oa = null; // これで属性 a は他のオブジェクトから参照されなくなり、オブジェクトはリサイクルできるようになります


制限: ループ

このアルゴリズムには制限があります。オブジェクトが別のオブジェクトを参照し、循環参照が形成されると、それらが不要になった場合でも、ガベージ コレクターはそれらを再利用しません。

コードをコピー コードは次のとおりです。

function f(){
var o = { };
var o2 = {};
o.a = o2; // o2 は o2
を参照します。 o2.a = o; // o2 は o

を参照します。 azerty";
}

f();
// 2 つのオブジェクトが作成され、相互参照を形成します
// 関数呼び出しが終了した後、それらは関数スコープから出ません。使用されませんが、解放されません
// これは、オブジェクトが参照されている限り、ガベージ コレクションを実行できないと参照カウント アルゴリズムが判断するためです

現実
ie6 および 7 の例では、DOM オブジェクトに対して参照カウント アルゴリズムが使用されており、メモリ リークの問題が発生します。

コードをコピー コードは次のとおりです。

var div = document.createElement("div ");
div.onclick = function(){
doSomething();
}; // div は click 属性を通じてイベント ハンドラーを参照します
// div 変数がアクセスされたときイベント ハンドラー関数は循環参照を形成し、両方のオブジェクトがリサイクルされなくなり、メモリ リークが発生します

タグ - アルゴリズムのクリア

彼は、「不要になったオブジェクト」と「到達不能なオブジェクト(オブジェクト到達不能)」という概念を導入しました。このアルゴリズムは、一連のルート オブジェクトが存在することを前提としています (JavaScript のルート オブジェクトはグローバル オブジェクトです)。ガベージ コレクターはルート オブジェクトから開始し、参照するすべてのオブジェクトを走査し、次に、ルート オブジェクトを走査します。参照オブジェクトによって参照されるオブジェクトなど。このアプローチを使用すると、ガベージ コレクターはアクセス可能なすべてのオブジェクトを取得し、アクセスできないオブジェクトを再利用できます。

このアルゴリズムは、0 によって参照されるオブジェクトがアクセス不可能なオブジェクトとして設定されると同時に、循環参照によって引き起こされる問題も回避します。

2012 年現在、ほとんどの最新ブラウザはこの「マーク アンド スイープ」ガベージ コレクターを使用しています。 JavaScript のガベージ コレクション (世代別/増分/同時/並列ガベージ コレクション) の分野は、ここ数年でそれに関連するアルゴリズムが改良されてきましたが、ガベージ コレクションのアルゴリズムそのもの (マークスイープ アルゴリズム) や「オブジェクトかどうかを判断する方法」は、はもう必要ありません」は改善されていません。

生理周期はもう問題ではありません

最初の例では、関数呼び出しが終了すると、これら 2 つのオブジェクトはグローバル オブジェクトによって参照されなくなり、グローバル オブジェクトによって参照されるオブジェクトからも参照されなくなります。したがって、これらは JavaScript ガベージ コレクターによってアクセス不可能なオブジェクトとしてマークされます。 2 番目の例でも同じことが起こり、div とイベント ハンドラーがガベージ コレクターによってアクセス不能としてマークされると、それらは解放されます。

制限事項: オブジェクトは明示的にアクセス不可としてマークする必要があります

このマーキング方法には制限がありますが、プログラミングでこの方法に触れたことがないため、ガベージ コレクション関連のコンテンツについてはほとんど気にしません。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

Video Face Swap

Video Face Swap

完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

C++ オブジェクトのレイアウトはメモリに合わせて調整され、メモリの使用効率が最適化されます。 C++ オブジェクトのレイアウトはメモリに合わせて調整され、メモリの使用効率が最適化されます。 Jun 05, 2024 pm 01:02 PM

C++ オブジェクト レイアウトとメモリ アライメントにより、メモリ使用効率が最適化されます。 オブジェクト レイアウト: データ メンバーは宣言の順序で格納され、スペース使用率が最適化されます。メモリのアライメント: アクセス速度を向上させるために、データがメモリ内でアライメントされます。 alignas キーワードは、キャッシュ ラインのアクセス効率を向上させるために、64 バイトにアライメントされた CacheLine 構造などのカスタム アライメントを指定します。

C++ メモリ管理: カスタム メモリ アロケータ C++ メモリ管理: カスタム メモリ アロケータ May 03, 2024 pm 02:39 PM

C++ のカスタム メモリ アロケータを使用すると、開発者は必要に応じてメモリ割り当て動作を調整できます。カスタム アロケータを作成するには、std::allocator を継承し、allocate() 関数と deallocate() 関数を書き直す必要があります。実際の例としては、パフォーマンスの向上、メモリ使用量の最適化、特定の動作の実装などが挙げられます。使用する場合は、メモリの解放を避けること、メモリのアライメントを管理すること、ベンチマーク テストを実行することなどに注意する必要があります。

大規模なコード ベースにおける C++ 関数のメモリ割り当てと破棄のベスト プラクティス 大規模なコード ベースにおける C++ 関数のメモリ割り当てと破棄のベスト プラクティス Apr 22, 2024 am 11:09 AM

C++ 関数のメモリ割り当てと破棄のベスト プラクティスには、静的メモリ割り当てにローカル変数を使用することが含まれます。動的メモリ割り当てにはスマート ポインタを使用します。メモリはコンストラクターで割り当てられ、デストラクターで破棄されます。複雑なメモリ シナリオにはカスタム メモリ マネージャーを使用します。例外処理を使用してリソースをクリーンアップし、例外が発生したときに割り当てられたメモリが確実に解放されるようにします。

C++ 関数のメモリ割り当てと破棄のための拡張機能と高度なテクニック C++ 関数のメモリ割り当てと破棄のための拡張機能と高度なテクニック Apr 22, 2024 pm 05:21 PM

C++ 関数のメモリ管理は、次のような拡張機能と高度なテクノロジを提供します。 カスタム アロケータ: ユーザーが独自のメモリ割り当て戦略を定義できるようにします。 Placementnew と Placementdelete: オブジェクトを特定のメモリ位置に割り当てる必要がある場合に使用されます。高度なテクノロジ: メモリ プール、スマート ポインタ、および RAII により、メモリ リークを削減し、パフォーマンスを向上させ、コードを簡素化します。

マルチスレッド環境における C++ メモリ管理の課題と対策? マルチスレッド環境における C++ メモリ管理の課題と対策? Jun 05, 2024 pm 01:08 PM

マルチスレッド環境では、C++ メモリ管理はデータ競合、デッドロック、メモリ リークなどの課題に直面します。対策には次のものが含まれます: 1. ミューテックスやアトミック変数などの同期メカニズムの使用、 2. ロックフリーのデータ構造の使用、 4. (オプション) ガベージ コレクションの実装。

C++ メモリ管理はオペレーティング システムおよび仮想メモリとどのように対話しますか? C++ メモリ管理はオペレーティング システムおよび仮想メモリとどのように対話しますか? Jun 02, 2024 pm 09:03 PM

C++ メモリ管理はオペレーティング システムと対話し、オペレーティング システムを通じて物理メモリと仮想メモリを管理し、プログラムにメモリを効率的に割り当ておよび解放します。オペレーティング システムは物理メモリをページに分割し、必要に応じてアプリケーションによって要求されたページを仮想メモリから取得します。 C++ は、new 演算子と delete 演算子を使用してメモリの割り当てと解放を行い、オペレーティング システムからメモリ ページを要求し、それらをそれぞれ返します。オペレーティング システムが物理メモリを解放すると、使用量の少ないメモリ ページが仮想メモリにスワップされます。

PHP 関数でのメモリ使用量を管理するにはどうすればよいですか? PHP 関数でのメモリ使用量を管理するにはどうすればよいですか? Apr 26, 2024 pm 12:12 PM

PHP 関数でのメモリ使用量を管理するには、不要な変数の宣言を回避し、未使用の変数を解放し、ループと条件を最適化します (無限ループの回避やインデックス付き配列の使用など)。

golang 関数と goroutine のメモリ管理 golang 関数と goroutine のメモリ管理 Apr 25, 2024 pm 03:57 PM

Go の関数のメモリは値によって渡され、元の変数には影響しません。 Goroutine はメモリを共有し、Goroutine が実行を完了するまで、割り当てられたメモリは GC によって再利用されません。メモリ リークは、完了した Goroutine 参照を保持するか、グローバル変数を使用するか、静的変数を回避することによって発生する可能性があります。リークを回避するには、チャネルを通じてゴルーチンをキャンセルし、静的変数を避け、defer ステートメントを使用してリソースを解放することをお勧めします。

See all articles