JVMシリーズのメモリモデルの詳細な説明

JVMシリーズのメモリモデルの詳細な説明

1. メモリモデルとランタイムデータ領域

この章では、Java 仮想マシンのメモリ モデルについて紹介します。JVM ランタイム データベースは仕様であり、JVM メモリ モデルはこの仕様の実装であると理解できます。

Java 仮想マシンはヒープ領域とメソッド領域にデータを格納することに重点を置いているため、この章ではこれら 2 つの側面についても詳しく説明します。ヒープとメソッド領域は共有メモリですが、Java 仮想マシン スタック、ネイティブ メソッド スタック、およびプログラム カウンターはスレッド専用です。

2. マインドマップと凡例

1 つは非ヒープ領域 (メソッド領域) であり、一般に「永続世代」とも呼ばれます。もう 1 つはヒープ領域で、若い領域と古い領域に分かれています。若い領域は 2 つの部分に分かれており、1 つは Eden 領域、もう 1 つは Survivor 領域 (S0+S1) です。S0 領域は From 領域とも呼ばれ、S1 は To 領域とも呼ばれます。

3. オブジェクトはJVMからスペースを申請する

4. サバイバーゾーンはなぜ必要なのでしょうか?

なぜ生存者エリアが必要なのでしょうか? エデンで十分ではないのですか?

Survivor 領域を設計しないと仮定すると、Eden 領域で MinorGC が実行され、オブジェクトは直接 Old 領域に送られます。このようにすると、Old 領域はすぐにいっぱいになります。Old 領域がいっぱいになると、FullGC が実行されます (Old 領域では MajorGC が実行され、通常は MinorGC が伴います)。FullGC は非常に時間がかかるため、Survivor 領域を設計する目的は、Old 領域に送られるオブジェクトの数を減らすことです。過渡的な Survivor 領域があります。

補足: マイナーGC: 新世代
メジャーGC: 旧世代
フルGC: 新世代 + 旧世代
エデン:S1:S2は8:1:1

5. なぜ 2 つのサバイバー ゾーンが必要なのですか?

2 つの Survivor 領域を必要とする目的は、メモリの断片化を回避することです。なぜそんなことを言うのですか?
Survivor スペースが 1 つだけ設計されていると仮定すると、Eden スペースがいっぱいになると、Minor GC が実行され、Eden スペース内の生き残ったオブジェクトが Survivor スペースに移動されます。次に Eden スペースがいっぱいになると、問題が発生します。Minor GC によって、Eden スペースのオブジェクトが Survivor スペースに強制的に移動され、オブジェクトが占有するメモリが不連続になります。

6. 検証例

ヒープメモリオーバーフロー

lombok.Data をインポートします。
org.springframework.web.bind.annotation.GetMapping をインポートします。
org.springframework.web.bind.annotation.RestController をインポートします。

java.util.ArrayList をインポートします。
java.util.List をインポートします。

@レストコントローラ
パブリッククラスHeapController{

    リスト<Foo> リスト = 新しい ArrayList<Foo>();
    @GetMapping(値 = {"ヒープ"})
    パブリック文字列ヒープテスト() {
        (真)の間{
            リストに追加します(新しいFoo());
        }
    }


    @データ
    クラスFoo {
        文字列 str;
    }
}

インターフェースにアクセスすると、メモリ オーバーフローが発生します。

java.lang.OutOfMemoryError: Java ヒープ スペース

...

パラメータを設定できます: 例: -Xms64M -Xmx512M

メソッド領域のメモリオーバーフロー

asm、maven 構成の使用:

<依存関係>
  <groupId>アセンブラ</groupId>
  <artifactId>アセンブリ</artifactId>
  <バージョン>3.3.1</バージョン>
</依存関係>

コードを記述し、メソッド領域にクラス情報を追加します。コンピュータのパフォーマンスが十分でない場合は、このコードを実行しないでください。コンピュータが再起動する原因になりやすいです。メモリを消費しすぎる場合は、ループの数を減らすこともできます。

org.objectweb.asm.ClassWriter をインポートします。
org.objectweb.asm.MethodVisitor をインポートします。
org.objectweb.asm.Opcodes をインポートします。

java.util.ArrayList をインポートします。
java.util.List をインポートします。

パブリッククラスMyMetaspaceはClassLoaderを拡張します{

  パブリック静的リスト<Class<?>>createClasses() {
    List<Class<?>> classes = new ArrayList<Class<?>>();
    (int i = 0; i < 10000000; ++i) の場合 {
      クラスライター cw = 新しいクラスライター(0);
      cw.visit(Opcodes.V1_1, Opcodes.ACC_PUBLIC, "クラス" + i, null,
              "java/lang/Object", null);
      MethodVisitor mw = cw.visitMethod(Opcodes.ACC_PUBLIC, "<init>",
              "()V", null, null);
      mw.visitVarInsn(Opcodes.ALOAD, 0);
      mw.visitMethodInsn(Opcodes.INVOKESPECIAL、"java/lang/Object"、
              "<init>", "()V");
      mw.visitInsn(オペコード.RETURN);
      mw.visitMaxs(1, 1);
      mw.visitEnd();
      MyMetaspace テスト = 新しい MyMetaspace();
      バイト[]コード = cw.toByteArray();
      クラス<?> exampleClass = test.defineClass("Class" + i, code, 0,
              コードの長さ);
      クラスを追加します(exampleClass);
    }
    クラスを返します。
  }
}

メソッド領域テストインターフェース:

com.example.jvm.jvmexceptionexample.asm.MyMetaspace をインポートします。
org.springframework.web.bind.annotation.GetMapping をインポートします。
org.springframework.web.bind.annotation.RestController をインポートします。

java.util.ArrayList をインポートします。
java.util.List をインポートします。

@レストコントローラ
パブリッククラスNonHeapController {

    リスト<Class<?>> list = 新しい ArrayList<Class<?>>();

    @GetMapping(値 = {"/noheap"})
    パブリック文字列noheap() {
        (真)の間{
            リストにすべてを追加します(MyMetaspace.createClasses());
        }
    }

}

java.lang.OutOfMemoryError: メタスペース

java.lang.ClassLoader.defineClass1(ネイティブメソッド) ~[na:1.8.5_54]

解決策: Metaspace のサイズを設定します (例: -XX:MetaspaceSize=64M -XX:MaxMetaspaceSize=512M)

Java 仮想マシン スタック

前に学んだように、Java 仮想マシン スタックはスタック フレームの形式で保存されます。1 つのメソッドは 1 つのスタック フレームに対応し、キュー モードでスタックにプッシュされます。したがって、プログラムが Java 仮想マシン スタックに問題を引き起こすかどうかをテストする場合は、再帰メソッドを使用してテストできます。

org.springframework.web.bind.annotation.GetMapping をインポートします。
org.springframework.web.bind.annotation.RestController をインポートします。

@レストコントローラ
パブリッククラスStackController{

    パブリック静的ロングカウント = 0;

    パブリック静的void add(long i) {
        カウント++;
        追加(i);
    }

    @GetMapping(値 = {"スタック"})
    パブリックボイドスタック(){
        追加(1);
    }

}

StackOverflow、スタックオーバーフロー例外:

java.lang.StackOverflowError: null

com.example.jvm.jvmexceptionexample.controller.StackController.add(StackController.java:14) で ~[classes/:na]

解決策: -Xss256k を設定します: 各スレッドのスタック サイズを設定します。 JDK 5 以降では、各スレッドのスタック サイズは 1M になり、それ以前は、各スレッドのスタック サイズは 256K でした。

以上がJVMシリーズのメモリモデルの詳細説明の内容です。JVMメモリモデルのメモリ構造の詳細については、123WORDPRESS.COMの他の関連記事に注目してください。

以下もご興味があるかもしれません:
  • JavaメモリモデルとJVMランタイムデータ領域の違いの詳細な説明
  • グラフィカル JVM メモリ モデル
  • JVM メモリ モデルの知識ポイントのまとめ
  • JVM メモリ モデルを理解するのに役立つ記事

<<:  Ubuntu 18.04 に mysql5.7.23 をインストールするチュートリアル

>>:  Windows 上で Zookeeper サーバーを構築するチュートリアル

推薦する

CSS3でハートを描く

成果を達成する要件/機能: CSS + HTML を使用してハートを描く方法。分析:正方形と 2 つ...

Vue ミックスインの詳しい説明

目次ローカルミックスイングローバル ミックスイン要約するローカルミックスイン <テンプレート&...

Vue.js スタイルレイアウト Flutter ビジネス開発共通スキル

シャドウスタイルにおけるフラッターとCSSの対応UIによって指定されたCSSスタイル 幅: 75px...

この記事ではMySQLのNULLについて説明します。

目次序文MySQL の NULL 2 NULL占有長3 NULL値の比較4SQLはNULL値を処理す...

MySQL における tinyint と int の違いの詳細な説明

質問: int(1) と tinyint(1) の違いは何ですか?このような設計では、いずれにしても...

CentOS7 systemdにカスタムシステムサービスを追加する方法

システムド: CentOS 7のサービスsystemctlスクリプトは、/usr/lib/syste...

画像を読み込むための JavaScript キャンバス

この記事では、画像を読み込むためのJavaScriptキャンバスの具体的なコードを参考までに紹介しま...

MySQL における識別子の大文字と小文字の区別の問題の詳細な分析

MySQL では、テーブル名の大文字と小文字の区別の問題が発生する可能性があります。実際、これはプラ...

ドラッグ効果を実現するための純粋なCSSコード

目次1. ドラッグ効果の例2. CSS実装の原則3. CSS実装の詳細4. CSSレイアウト1. 固...

MySQL は、現在のデータ テーブル内のすべての時間に対して指定された時間間隔を増加または減少させます (推奨)

DATE_ADD() 関数は、指定された時間間隔を日付に追加します。現在のテーブル内のすべてのデー...

相対幅と絶対幅が競合する場合の HTML+CSS div ソリューション

相対幅と絶対幅が競合する場合のdivソリューション概要: 一般的に、絶対幅を使用する場合は px を...

MySQL における一般的な高度な SQL ステートメント

MySQL 高度な SQL ステートメント kgc を使用します。 テーブルlocation(Reg...

Apache SkyWalkingのセルフモニタリングを素早く有効にする方法を説明します

1. Prometheusテレメトリデータを有効にするデフォルトでは、テレメトリは次のように無効にな...

Docker は MySQL をインストールし、中国語の文字化けの問題を解決します

目次1. MySQLイメージを取得する2. ダウンロードが完了したか確認する3. MySQLはローカ...

CSS でのフレックスレイアウトの詳細な説明

フレックス レイアウトは、エラスティック レイアウトとも呼ばれます。任意のコンテナーをフレックス レ...