ホームページ Java &#&チュートリアル JAVA 仮想マシン (JVM) の詳細な紹介 (7​​) - JVM の最適化

JAVA 仮想マシン (JVM) の詳細な紹介 (7​​) - JVM の最適化

Aug 24, 2019 pm 04:41 PM
jvm

JAVA 仮想マシン (JVM) の詳細な紹介 (7​​) - JVM の最適化

この図を例にとると、.java から .class へのコンパイル処理が、.class からマシンコードへの解釈処理になります。これらは以下で個別に最適化されます。最適化プロセスにおいて、コンパイル段階の最適化は主にフロントエンドコンパイラの最適化であり、実行段階の最適化は主にジャストインタイムコンパイラの最適化です。

JAVA 仮想マシン (JVM) の詳細な紹介 (7​​) - JVM の最適化

#コンパイラの最適化

##コンパイル プロセス

JAVA 仮想マシン (JVM) の詳細な紹介 (7​​) - JVM の最適化

#上記は javac コンパイル プロセスの図であり、以下は javac コンパイル プロセスのメイン コードです。

#次の手順について詳しく説明しますJAVA 仮想マシン (JVM) の詳細な紹介 (7​​) - JVM の最適化1. シンボル テーブルの解析と入力

語彙分析

ソース コードの文字ストリームをトークンのコレクションに変換します。トークンは、コンパイル プロセスにおける最小の要素 (a、=、b、int など) です。

構文解析

トークン シーケンスに基づいて抽象構文ツリーを構築します。将来的には、コンパイラは基本的にソース コード ファイルに対して動作しなくなり、その後の動作は抽象構文ツリーに基づいて行われるようになります。抽象構文ツリーは、プログラム コードの構文構造を記述するために使用されるツリー表現です。ノードは、修飾子、戻り値など、コード内の構文構造を表します。

シンボル テーブルの入力

シンボル テーブルは、シンボル アドレスとシンボル情報のセットで構成されるテーブルで、コンパイルのさまざまな段階で使用されます。たとえば、意味解析では意味チェックや中間コード生成に使用され、ターゲットコード生成段階ではアドレス割り当ての基礎として使用されます。

2. アノテーション プロセッサ

この部分は、コンパイル中にアノテーションを処理するプラグイン アノテーション プロセッサのプロセスです。構文ツリーを変更できます。変更後、コンパイラは再処理のために上記の最初のステップに戻ります。各ループは Round と呼ばれ、上図のループバック プロセスに相当します。
3. 意味解析とバイトコード生成

構文解析後、生成された構文ツリーは正しい構造を持つソース プログラムを抽象化したものですが、ソースが正しい構造であるという保証はありません。プログラムは論理的です。セマンティック分析のタスクは、構造的に正しいソース プログラムのコンテキスト依存的な性質を調べることです。たとえば、次のコードのエラーは、セマンティック分析段階でのみチェックできます。

boolean a=false;
char b=2;
int c=a+b
ログイン後にコピー
このステージには次の 4 つのステップが含まれます。

注釈チェック

変数が使用前に宣言されているかどうか、変数と変数の間のデータ割り当て、型が一致するかどうかなど。 a=1 2 を a=3 に変える定数折りも存在します。したがって、コード内の a=1 2 および a=3 によって、プログラムの実行中に CPU 命令の操作量が増加することはありません。

データと制御フローの分析

プログラムのローカル変数に使用前に値が割り当てられているかどうか、メソッドの各パスに戻り値があるかどうかを確認します。すべてがチェックされているかどうか、例外が正しく処理されるかどうか、およびその他の問題が発生するかどうか。クラスロード時のデータ・制御フロー解析もあり、目的は基本的に同じですが、検証範囲が異なり、一部の検証項目はコンパイル時またはランタイム時のみ実行できます。

構文シュガーを理解する

構文シュガーとは、コンピューター言語に特定の構文を追加することです。言語の機能には影響しませんが、可読性を向上させることができます。プログラムの。構文シュガーには、ジェネリック、自動アンボックス化などが含まれます。これらの構文は仮想マシン ランタイムではサポートされていないため、コンパイル フェーズ中に基本構文構造に戻ります。このプロセスは、構文シュガーのデコードと呼ばれます。

バイトコード生成

これまでの手順で生成した情報(構文ツリー、シンボルテーブル)をバイトコードに変換してディスクに書き込み、小さなAを追加して変換します。コードの量。たとえば、文字列の追加操作を StringBuffer または StringBuilder の append() 操作に置き換えます。

この時点で、Class ファイルが生成されます。

语法糖

语法糖是java中添加某种语法,对语言的功能没有影响,但是可以增加程序的可读性。包括泛型、内部类、枚举类等。

1、泛型与类型擦除
泛型可用于类、接口和方法的创建中,用于对放入集合元素的类型的约束。泛型只在程序源码中存在,在编译阶段有解语法糖的步骤,所以在.Class文件中,已经变为了原来的原生类型了。这个过程叫做类型擦除。
泛型擦除前:

public static void main(String[] args){
    Map<String,String> map=new HashMap<>();
    map.put("姓名","小明");
    map.put("性别","男");
    sout(map.get("姓名"));
    sout(map.get("性别"));
}
ログイン後にコピー

泛型擦除后:

public static void main(String[] args){
    Map map=new HashMap();
    map.put("姓名","小明");
    map.put("性别","男");
    sout((String)map.get("姓名"));
    sout((String)map.get("性别"));
}
ログイン後にコピー

所以ArrayList和ArrayList在运行期时是同一个类。

2、自动拆装箱、循环遍历
这些是java中使用最多的语法糖。编译前:

public static void main(String[] args){
      List<Integer> list=Arrays.asList(1,2,3,4);
      int sum=0;
      for(int i:list){
              sum +=i;
      }
      System.out.println(sum);
}
ログイン後にコピー

编译后:

public static void main(String[] args){
      List list=Arrays.asList(new Integer[] {
                Integer.valueOf(1),
                Integer.valueOf(2),
                Integer.valueOf(3),
                Integer.valueOf(4)});
      int sum=0;
      for(Iterator localIterator=list.iterator();localIterator.hasNext();){
                   int i=((Integer)localIterator.next()).intValue();
                   sum +=i;
      }
      System.out.println(sum);
}
ログイン後にコピー

可见,自动拆装箱在编译后被转化为了对应的包装和还原方法,如Integer.valueOf()和Integer.intValue()。
遍历循环则把代码还原为了迭代器的实现。

3、条件编译
根据布尔常量值的真假,编译器会把分支中不成立的代码块消除掉。

public static void main(String[] args){
      if(true){
            sout("block 1");
      }else{
           sout("block 2");
     }
}
ログイン後にコピー

编译后,代码变为:

public static void main(String[] args){
     sout("block 1");
}
ログイン後にコピー

运行期优化

一般情况下,我们将.java编译成.class,.class再解释成机器码。但是也有特殊的情况。有些代码调用比较频繁,比如某个方法或代码块的运行特别频繁,为了提高程序的执行效率,在运行时,虚拟机会把这个代码直接编译成机器码,并进行各种层次的优化。这样的代码称为热点代码。完成这个任务的编译器被称为即时编译器。但是其并不是虚拟机必需的部分。

JAVA 仮想マシン (JVM) の詳細な紹介 (7​​) - JVM の最適化

即时编译器的概述

(1)为什么虚拟机要使用解释器和编译器并存的架构?

虚拟机里包含着解释器和编译器。当程序需要迅速启动和执行的时候,解释器可以首先发挥作用,省去编译的时间,立即执行。在程序运行后,随着时间的推移,编译器逐渐发挥作用,把越来越多的代码编译成本地代码之后,可以获取更高的执行效率。

当程序运行环境中内存资源限制较大,可以使用解释执行节约内存,反之可以使用编译来提升效率。

(2)为什么虚拟机要实现两个不同的即时编译器?

虚拟机中内置了两个即时编译器,分别为Client Compiler和Server Compiler,又称为C1和C2。

默认只使用其中的一个,至于选择哪个,取决于虚拟机会根据自身版本和宿主机器的硬件性能自动选择运行模式。用户也可以使用“-client”、“-server”进行指定。

(3)程序何时使用解释器执行?何时使用编译器执行?

虚拟机有一个分层编译策略。

第0层:程序解释执行,解释器不开启性能监控功能,可触发第1层编译

第1层:也称为C1编译,将字节码编译为本地代码,进行简单、可靠的优化,如有必要将加入性能监控的逻辑。

第2层:也称为C2编译,将字节码编译为本地代码,但是会启用一些编译耗时较长的优化,甚至会根据性能监控信息进行一些不可靠的激进优化。

(4)哪些程序代码会被编译为本地代码?如何编译为本地代码?

热点代码包括如下两类,其均把整个方法作为编译对象。

a、被多次调用的方法

b、被多次执行的循环体

热点探测是用来判断一段代码是否为热点代码,其方式有两种:

a、基于采样

b、基于计数器。HotSpot使用的是这种。它为每个方法准备了两类计数器:统计方法被调用次数的方法调用计数器和统计一个方法中循环体代码执行次数的回边计数器。

(5)如何从外部观察及时编译器的编译过程和编译结果?

可以使用 -xx:+PrintCompilation 查看哪些方法被即时编译器编译了。

优化技术有哪些?

虚拟机的即时编译器在生成代码时,采用了如下的代码优化技术。

(1)公共子表达式消除

如果一个表达式E已经计算过了,那如果再次出现E时就不会再对它进行计算。比如:

int d=(a*b)*12+c+(c+b*a)
ログイン後にコピー

如果这段代码交给javac编译器,则不会进行任何优化。如果交给即时编译器,会被进行如下步骤的优化:

第一步:消除公共子表达式

int d=E*12+c+(c+E)
ログイン後にコピー

第二步:代数化简:

int d=E*13+c*2
ログイン後にコピー

(2)数组边界检查消除

数组边界检查是什么?
如果有一个数组foo[],在java语言中访问数组元素foo[i]的时候,系统将会自动进行上下界的范围检查,检查i是否满足0≤i≤foo.length这个条件。

那怎么进行消除呢?
a、把运行期检查提到编译期完成。如foo[3],只要在编译期根据数据流分析来确定foo.length的值,并判断下标“3”没有越界,执行的时候就不用判断了。
b、隐式异常处理。这种思路通常用于空指针检查和算符运算中除数为零的情况。

if(foo!=null){
  return foo.value;
}else{
  throw new NullPointException();
}
ログイン後にコピー

被隐式异常处理优化后,变为如下代码:

try{
  return foo.value;
}catch(segment_fault){
  uncommon_trap();
}
ログイン後にコピー

除了数组边界检查消除,还有自动装箱消除、安全点消除、消除反射等。

(3)方法内联

把目标方法的代码“复制”到发起调用的方法之中,避免发生真实的方法调用。

public int add(int x1, int x2, int x3, int x4) {  
        return add1(x1, x2) + add1(x3, x4);  
 }  

public int add1(int x1, int x2) {  
        return x1 + x2;  
    }
ログイン後にコピー

运行一段时间后JVM会把add1方法去掉,并把代码翻译成:

public int add(int x1, int x2, int x3, int x4) {  
        return x1 + x2 + x3 + x4;  
}
ログイン後にコピー

(4)逃逸分析

当一个对象在方法中被定义后,它可能被外部方法所引用,比如作为调用参数传递到其它方法中,这称为方法逃逸。同理,如果被外部线程访问到,它就称为线程逃逸。

对变量进行相应分析就叫做逃逸分析。如果能证明别的方法或线程无法通过任何途径访问到这个对象,则可以为这个变量进行一些优化。

优化的手段有栈上分配、同步消除、标量替换等。以同步消除为例,如果逃逸分析能够确定一个变量不会逃逸出线程,即无法被其它线程访问到,那对这个变量实施的同步措施就可以消除掉了。

以上内容便是关于JAVA虚拟机中JVM优化的全部介绍,更多相关问题请访问PHP中文网:JAVA视频教程

以上がJAVA 仮想マシン (JVM) の詳細な紹介 (7​​) - JVM の最適化の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、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)

非常に実用的な分散型 JVM 監視ツールです。 非常に実用的な分散型 JVM 監視ツールです。 Aug 15, 2023 pm 05:15 PM

このプロジェクトは、開発者が複数のリモート ホスト JVM をより速く監視できるように設計されています。プロジェクトが Spring Boot の場合、統合は非常に簡単です。jar パッケージを導入するだけです。Spring Boot でない場合でも、落胆する必要はありません。 Spring Boot プログラムを手早く初期化し、自分で導入する Jar パッケージで十分です

JVM コマンドラインパラメータの詳細説明: JVM の動作を制御する秘密兵器 JVM コマンドラインパラメータの詳細説明: JVM の動作を制御する秘密兵器 May 09, 2024 pm 01:33 PM

JVM コマンド ライン パラメータを使用すると、JVM の動作をきめ細かいレベルで調整できます。共通パラメータは次のとおりです。 Java ヒープ サイズの設定 (-Xms、-Xmx) 新しい世代サイズの設定 (-Xmn) パラレル ガベージ コレクタの有効化 (-XX:+UseParallelGC) Survivor 領域のメモリ使用量の削減 (-XX: -ReduceSurvivorSetInMemory) 冗長性の削除 ガベージ コレクションの削除 (-XX:-EliminateRedundantGCs) ガベージ コレクション情報の印刷 (-XX:+PrintGC) G1 ガベージ コレクターの使用 (-XX:-UseG1GC) ガベージ コレクションの最大休止時間の設定 (-XX:MaxGCPau)

JVMメモリ管理のポイントと注意事項 JVMメモリ管理のポイントと注意事項 Feb 20, 2024 am 10:26 AM

JVM のメモリ使用量を使いこなすためのポイントと注意点 JVM (JavaVirtualMachine) は Java アプリケーションが動作する環境であり、最も重要なのは JVM のメモリ管理です。 JVM メモリを適切に管理すると、アプリケーションのパフォーマンスが向上するだけでなく、メモリ リークやメモリ オーバーフローなどの問題も回避できます。この記事では、JVM メモリ使用の重要なポイントと考慮事項を紹介し、いくつかの具体的なコード例を示します。 JVM メモリ パーティション JVM メモリは主に次の領域に分割されます。 ヒープ (ヒープ)

Java エラー: JVM メモリ オーバーフロー エラー、対処方法と回避方法 Java エラー: JVM メモリ オーバーフロー エラー、対処方法と回避方法 Jun 24, 2023 pm 02:19 PM

Java は人気のあるプログラミング言語ですが、Java アプリケーションの開発中に、JVM メモリ オーバーフロー エラーが発生する場合があります。通常、このエラーによりアプリケーションがクラッシュし、ユーザー エクスペリエンスに影響を与えます。この記事では、JVM メモリ オーバーフロー エラーの原因と、そのようなエラーに対処および回避する方法について説明します。 JVMメモリオーバーフローエラーとは何ですか? Java 仮想マシン (JVM) は、Java アプリケーションの実行環境です。 JVM では、メモリはヒープ、メソッド領域、スタックなどの複数の領域に分割されます。ヒープは作成されたオブジェクトを保存するために使用されます

JVM仮想マシンの機能と原理の分析 JVM仮想マシンの機能と原理の分析 Feb 22, 2024 pm 01:54 PM

JVM 仮想マシンの機能と原理の分析の紹介: JVM (JavaVirtualMachine) 仮想マシンは、Java プログラミング言語の中核コンポーネントの 1 つであり、Java の最大のセールス ポイントの 1 つです。 JVM の役割は、Java ソース コードをバイトコードにコンパイルし、これらのバイトコードを実行することです。この記事では、JVM の役割とその仕組みを紹介し、読者の理解を深めるためにいくつかのコード例を示します。機能: JVM の主な機能は、さまざまなプラットフォーム上での Java プログラムの移植性の問題を解決することです。

JVM ヒープ メモリ サイズを効率的に調整するにはどうすればよいですか? JVM ヒープ メモリ サイズを効率的に調整するにはどうすればよいですか? Feb 18, 2024 pm 01:39 PM

JVM メモリ パラメータ設定: ヒープ メモリ サイズを合理的に調整するにはどうすればよいですか? Java アプリケーションでは、JVM はメモリの管理を担当する主要なコンポーネントです。このうちヒープ メモリはオブジェクト インスタンスの保存に使用され、ヒープ メモリのサイズ設定はアプリケーションのパフォーマンスと安定性に重要な影響を与えます。この記事では、ヒープメモリサイズを合理的に調整する方法を具体的なコード例とともに紹介します。まず、JVM メモリに関する基本的な知識を理解する必要があります。 JVM のメモリは、ヒープ メモリ、スタック メモリ、メソッド領域などを含むいくつかの領域に分割されます。で

JVM が 32 ビットか 64 ビットかを確認する Java プログラム JVM が 32 ビットか 64 ビットかを確認する Java プログラム Sep 05, 2023 pm 06:37 PM

JVM が 32 ビットか 64 ビットかを確認する Java プログラムを作成する前に、まず JVM について説明します。 JVM は Java 仮想マシンであり、バイトコードの実行を担当します。これは Java ランタイム環境 (JRE) の一部です。 Java はプラットフォームに依存しませんが、JVM はプラットフォームに依存することは誰もが知っています。オペレーティング システムごとに個別の JVM が必要です。 Java ソース コードのバイトコードがあれば、JVM により任意のプラットフォームで簡単に実行できます。 Java ファイル実行のプロセス全体は次のとおりです。まず、Java ソース コードを .java 拡張子で保存し、コンパイラがそれを .class 拡張子を持つバイトコードに変換します。これはコンパイル時に発生します。さて、実行時に、J

JVM の動作原理の謎を解く: Java 仮想マシンの原理の詳細な探求 JVM の動作原理の謎を解く: Java 仮想マシンの原理の詳細な探求 Feb 18, 2024 pm 12:28 PM

JVM 原理の詳細な説明: Java 仮想マシンの動作原理を詳しく調べるには、特定のコード例が必要です。 1. はじめに Java プログラミング言語の急速な発展と広範な応用により、Java 仮想マシン (JavaVirtualMachine、JVM と呼ばれます) が登場しました。 )もソフトウェア開発には欠かせないものとなっています。 Java プログラムの実行環境として、JVM はクロスプラットフォーム機能を提供し、Java プログラムをさまざまなオペレーティング システムで実行できるようにします。この記事では、JVM の仕組みについて詳しく説明します。

See all articles