jit 方法内联_JIT编译器,内联和转义分析
jit 方法內聯
即時(JIT)
即時(JIT)編譯器是Java虛擬機的大腦。 JVM中對JIT編譯器的影響最大。
一會兒,讓我們退后一步,看看已編譯和未編譯語言的示例。
諸如Go,C和C ++之類的語言之所以稱為編譯語言,是因為它們的程序以二進制(編譯)代碼的形式分發,該代碼針對特定的CPU。
另一方面, 解釋語言如PHP和Perl。 只要機器具有解釋器,就可以在任何CPU上運行相同的程序源代碼。 解釋器在執行該行時將程序的每一行轉換為二進制代碼。
Java試圖在這里找到中間立場。 Java應用程序已編譯,但沒有被編譯為特定CPU的特定二進制文件,而是被編譯為bytecode 。 這為Java提供了與解釋語言無關的平臺。 但是Java不止于此。
在典型的程序中,只有一小部分代碼會頻繁執行,而應用程序的性能主要取決于這些代碼部分的執行速度。 這些關鍵部分被稱為應用程序的熱點 。
JVM執行特定代碼段的次數越多,有關它的信息就越多。 這使JVM可以做出明智/優化的決策,并將小的熱代碼編譯為CPU特定的二進制文件。 該過程稱為即時編譯(JIT) 。
現在,我們運行一個小程序,觀察JIT編譯。
public class App {public static void main(String[] args) {long sumOfEvens = 0;for(int i = 0; i < 100000; i++) {if(isEven(i)) {sumOfEvens += i;}}System.out.println(sumOfEvens);}public static boolean isEven(int number) {return number % 2 == 0;} }#### Run javac App.java && \ java -server \-XX:-TieredCompilation \-XX:+PrintCompilation \- XX:CompileThreshold=100000 App#### Output 87 1 App::isEven (16 bytes) 2499950000輸出告訴我們isEven方法已編譯。 我故意禁用了TieredCompilation,以僅獲取最常編譯的代碼。
JIT編譯的代碼將大大提高您的應用程序的性能。 要檢查嗎? 編寫一個簡單的基準代碼。
內聯
內聯是JIT編譯器進行的最重要的優化之一。 內聯將方法調用替換為方法的主體,以避免方法調用的開銷。
讓我們再次運行相同的程序,這次觀察內聯。
#### Run javac App.java && \ java -server \-XX:+UnlockDiagnosticVMOptions \-XX:+PrintInlining \-XX:-TieredCompilation App#### Output @ 12 App::isEven (16 bytes) inline (hot) 2499950000再次內聯將大大提高您的應用程序的性能。
轉義分析
轉義分析是一種技術,通過該技術,JIT編譯器可以分析新對象的使用范圍,并決定將其分配在Java堆還是方法堆棧上。 它還消除了對所有非全局轉義對象的鎖定
讓我們運行一個小程序,觀察垃圾回收。
public class App {public static void main(String[] args) {long sumOfArea = 0;for(int i = 0; i < 10000000; i++) {Rectangle rect = new Rectangle(i+5, i+10);sumOfArea += rect.getArea();}System.out.println(sumOfArea);}static class Rectangle {private int height;private int width;public Rectangle(int height, int width) {this.height = height;this.width = width;}public int getArea() {return height * width;}} }在此示例中,矩形對象被創建并且僅在循環內可用,它們的特征是NoEscape,它們將分配在堆棧上而不是堆上。 具體來說,這意味著將不會發生垃圾回收。
讓我們在沒有EscapeAnalysis的情況下運行程序。
#### Run javac App.java && \ java -server \-verbose:gc \-XX:-DoEscapeAnalysis App#### Output [GC (Allocation Failure) 65536K->472K(251392K), 0.0007449 secs] [GC (Allocation Failure) 66008K->440K(251392K), 0.0008727 secs] [GC (Allocation Failure) 65976K->424K(251392K), 0.0005484 secs] 16818403770368如您所見,GC已啟動。 分配失敗意味著年輕一代中沒有剩余空間來分配對象。 因此,這是年輕GC的正常原因。
這次讓我們使用EscapeAnalysis運行它。
#### Run javac App.java && \ java -server \-verbose:gc \-XX:+DoEscapeAnalysis App#### Output 16818403770368這次沒有發生GC。 從根本上講,這意味著創建壽命短且作用域狹窄的對象不一定會引入垃圾。
默認情況下, DoEscapeAnalysis選項處于啟用狀態。 請注意,只有Java HotSpot Server VM支持此選項。
因此,我們所有人都應該避免過早的優化,而應專注于編寫更具可讀性/可維護性的代碼,并讓JVM來完成它的工作。
翻譯自: https://www.javacodegeeks.com/2015/12/jit-compiler-inlining-escape-analysis.html
jit 方法內聯
總結
以上是生活随笔為你收集整理的jit 方法内联_JIT编译器,内联和转义分析的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: (备案倒金款)
- 下一篇: jpa 关联实体的关联实体_JPA实体锁