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 サーバーを構築するチュートリアル

推薦する

Nginx で Basic Auth ログイン認証を設定する方法

nginx でファイルサーバーを構築することもありますが、これは一般に公開されていますが、サーバーが...

Linux sshのデフォルトのリモートポート番号を変更する6つの手順

Linux のデフォルトの ssh リモート ポートは 22 です。デフォルトのポートは、悪意のある...

MySQLはtruncateコマンドを使用してデータベース内のすべてのテーブルを素早くクリアします

1. まずSELECT文を実行して、すべての切り捨て文を生成します。ステートメント形式: selec...

MySql 学習ノートにおけるトランザクション分離レベルの詳細な説明

背景トランザクションについて話すとき、誰もがそれに精通している必要があります。MySQL データベー...

ウィンドウの中央にブロック要素の位置を設定する方法

ウィンドウの中央にブロック要素の位置を設定する方法ブロック要素をウィンドウの中央に配置する上記の方法...

CSS のサイズ単位についての簡単な説明

ブラウザの互換性はどんどん良くなってきています。モバイル端末は基本的にすべてWebKitベースです。...

Angularルーティングの基礎の詳細な説明

目次1. ルーティング関連オブジェクト2. ルーティングオブジェクトの場所3. ルーティング構成4....

CSS でよく発生する問題の整理 (ロゴのハッキング/コンテナの固定/画像の垂直方向の中央揃え)

1. IEブラウザモードハックロゴ1. CSSハックロゴコードをコピーコードは次のとおりです。 ie...

MySQLテーブルのテーブル構造を素早く変更する方法

MySQL テーブルのテーブル構造をすばやく変更する - 「MySQL 管理」から抜粋 ALTER ...

Windows 10 の Docker で countly-server を展開して実行するプロセス

私は最近countlyに触れて、慣れてきました。私は、必要に応じてcountlyのクラッシュプラグイ...

読めばわかるVueの8つのヒント

1. v-forでは常に:keyを使用するデータを操作する必要があるときにプログラムを一定かつ予測可...

Vue2.0の双方向データバインディング原則を手動で実装する

一言で言えば: データハイジャック (Object.defineProperty) + パブリッシュ...

フロントエンドアプリケーションのjenkins+gitlab+nginxデプロイメント

目次関連する依存関係のインストールドッカーDockerでJenkinsをインストールするDocker...

DockerコンテナにNFS共有ディレクトリをマウントする実装

以前、https://www.jb51.net/article/205922.htm で、Docke...