這篇文章將為大家詳細講解有關如何進行JVM虛擬機中Java的編譯期優(yōu)化與運行期優(yōu)化,文章內容質量較高,因此小編分享給大家做個參考,希望大家閱讀完這篇文章后對相關知識有一定的了解。
讓客戶滿意是我們工作的目標,不斷超越客戶的期望值來自于我們對這個行業(yè)的熱愛。我們立志把好的技術通過有效、簡單的方式提供給客戶,將通過不懈努力成為客戶在信息化領域值得信任、有價值的長期合作伙伴,公司提供的服務項目有:申請域名、網(wǎng)站空間、營銷軟件、網(wǎng)站建設、振安網(wǎng)站維護、網(wǎng)站推廣。
java編譯期優(yōu)化
java語言的編譯期其實是一段不確定的操作過程,因為它可以分為三類編譯過程:
1.前端編譯:把.java文件轉變?yōu)?/em>.class文件
2.后端編譯:把字節(jié)碼轉變?yōu)闄C器碼
3.靜態(tài)提前編譯:直接把.java文件編譯成本地機器代碼
從JDK1.3開始,虛擬機設計團隊就把對性能的優(yōu)化集中到了后端的即時編譯中,這樣可以讓那些不是由Javac產(chǎn)生的Class文件(如JRuby、Groovy等語言的Class文件)也能享受到編譯期優(yōu)化所帶來的好處
*Java中即時編譯在運行期的優(yōu)化過程對于程序運行來說更重要,而前端編譯期在編譯期的優(yōu)化過程對于程序編碼來說關系更加密切
早期編譯過程主要分為3個部分:1.解析與填充符號表過程:詞法、語法分析;填充符號表 2.插入式注解處理器的注解處理過程 3.語義分析與字節(jié)碼生成過程:標注檢查、數(shù)據(jù)與控制流分析、解語法糖、字節(jié)碼生成
Java語言中的泛型只在程序源碼中存在,在編譯后的字節(jié)碼文件中,就已經(jīng)替換成原來的原生類型了,并且在相應的地方插入了強制轉型代碼
泛型擦除前的例子 public static void main( String[] args ) { Mapmap = new HashMap (); map.put("hello","你好"); System.out.println(map.get("hello")); } 泛型擦除后的例子 public static void main( String[] args ) { Map map = new HashMap(); map.put("hello","你好"); System.out.println((String)map.get("hello")); }
自動裝箱、拆箱在編譯之后會被轉化成對應的包裝和還原方法,如Integer.valueOf()與Integer.intValue(),而遍歷循環(huán)則把代碼還原成了迭代器的實現(xiàn),變長參數(shù)會變成數(shù)組類型的參數(shù)。
然而包裝類的“==”運算在不遇到算術運算的情況下不會自動拆箱,以及它們的equals()方法不處理數(shù)據(jù)轉型的關系。
Java語言也可以進行條件編譯,方法就是使用條件為常量的if語句,它在編譯階段就會被“運行”:
public static void main(String[] args) { if(true){ System.out.println("block 1"); } else{ System.out.println("block 2"); } } 編譯后Class文件的反編譯結果: public static void main(String[] args) { System.out.println("block 1"); }
只能是條件為常量的if語句,這也是Java語言的語法糖,根據(jù)布爾常量值的真假,編譯器會把分支中不成立的代碼塊消除掉
Java程序最初是通過解釋器進行解釋執(zhí)行的,當程序需要迅速啟動和執(zhí)行時,解釋器可以首先發(fā)揮作用,省去編譯時間,立即執(zhí)行;當程序運行后,隨著時間的推移,編譯期逐漸發(fā)揮作用,把越來越多的代碼編譯成本地代碼,獲得更高的執(zhí)行效率。解釋執(zhí)行節(jié)約內存,編譯執(zhí)行提升效率。同時,解釋器可以作為編譯器激進優(yōu)化時的一個“逃生門”,讓編譯器根據(jù)概率選擇一些大多數(shù)時候都能提升運行速度的優(yōu)化手段,當激進優(yōu)化的假設不成立,則通過逆優(yōu)化退回到解釋狀態(tài)繼續(xù)執(zhí)行。
HotSpot虛擬機中內置了兩個即時編譯器,分別稱為Client Compiler(C1編譯器)和Server Compiler(C2編譯器),默認采用解釋器與其中一個編譯器直接配合的方式工作,使用哪個編譯器取決于虛擬機運行的模式,也可以自己去指定。若強制虛擬機運行與“解釋模式”,編譯器完全不介入工作,若強制虛擬機運行于“編譯模式”,則優(yōu)先采用編譯方式執(zhí)行程序,解釋器仍然要在編譯無法進行的情況下介入執(zhí)行過程。
分層編譯策略作為默認編譯策略在JDK1.7的Server模式虛擬機中被開啟,其中包括: 第0層:程序解釋執(zhí)行,解釋器不開啟性能監(jiān)控功能,可觸發(fā)第1層編譯; 第1層:C1編譯,將字節(jié)碼編譯成本地代碼,進行簡單可靠的優(yōu)化,如有必要將加入性能監(jiān)控的邏輯; 第2層:C2編譯,也是將字節(jié)碼編譯成本地代碼,但是會啟動一些編譯耗時較長的優(yōu)化,甚至會根據(jù)性能監(jiān)控信息進行一些不可靠的激進優(yōu)化。 實施分層編譯后,C1和C2將會同時工作,C1獲取更高的編譯速度,C2獲取更好的編譯質量,在解釋執(zhí)行的時候也無須再承擔性能監(jiān)控信息的任務。
在運行過程中會被即時編譯器編譯的“熱點代碼”有兩類: 1.被多次調用的方法:由方法調用觸發(fā)的編譯,屬于JIT編譯方式 2.被多次執(zhí)行的循環(huán)體:也以整個方法作為編譯對象,因為編譯發(fā)生在方法執(zhí)行過程中,因此成為棧上替換(OSR編譯) 熱點探測判定方式有兩種: 1.基于采樣的熱點探測:虛擬機周期性的檢查各個線程的棧頂,如果某個方法經(jīng)常出現(xiàn)在棧頂,則判定為“熱點方法”。(簡單高效,可以獲取方法的調用關系,但容易受線程阻塞或別的外界因素影響擾亂熱點探測) 2.基于計數(shù)的熱點探測:虛擬機為每個方法建立一個計數(shù)器,統(tǒng)計方法的執(zhí)行次數(shù),超過一定閾值就是“熱點方法”。(需要為每個方法維護計數(shù)器,不能直接獲取方法的調用關系,但是統(tǒng)計結果精確嚴謹)
HotSpot虛擬機使用的是第二種,它為每個方法準備了兩類計數(shù)器:方法調用計數(shù)器和回邊計數(shù)器,下圖表示方法調用計數(shù)器觸發(fā)即時編譯:
如果不做任何設置,執(zhí)行引擎會繼續(xù)進入解釋器按照解釋方式執(zhí)行字節(jié)碼,直到提交的請求被編譯器編譯完成,下次調用才會使用已編譯的版本。另外,方法調用計數(shù)器的值也不是一個絕對次數(shù),而是一段時間之內被調用的次數(shù),超過這個時間,次數(shù)就減半,這稱為計數(shù)器熱度的衰減。
下圖表示回邊計數(shù)器觸發(fā)即時編譯:
回邊計數(shù)器沒有計數(shù)器熱度衰減的過程,因此統(tǒng)計的就是絕對次數(shù),并且當計數(shù)器溢出時,它還會把方法計數(shù)器的值也調整到溢出狀態(tài),這樣下次進入該方法的時候就會執(zhí)行標準編譯過程。
虛擬機設計團隊幾乎把對代碼的所有優(yōu)化措施都集中在了即時編譯器之中,那么在編譯器編譯的過程中,到底做了些什么事情呢?下面將介紹幾種最有代表性的優(yōu)化技術:
公共子表達式消除
如果一個表達式E已經(jīng)計算過了,并且先前的計算到現(xiàn)在E中所有變量的值都沒有發(fā)生變化,那么E的這次出現(xiàn)就成為了公共表達式,可以直接用之前的結果替換。
例:int d = (c
b) 12 + a + (a + b
c) => int d = E 12 + a + (a + E)
數(shù)組邊界檢查消除
Java語言中訪問數(shù)組元素都要進行上下界的范圍檢查,每次讀寫都有一次條件判定操作,這無疑是一種負擔。編譯器只要通過數(shù)據(jù)流分析就可以判定循環(huán)變量的取值范圍永遠在數(shù)組長度以內,那么整個循環(huán)中就可以把上下界檢查消除,這樣可以省很多次的條件判斷操作。
另一種方法叫做隱式異常處理,Java中空指針的判斷和算術運算中除數(shù)為0的檢查都采用了這個思路:
if(foo != null){ return foo.value; }else{ throw new NullPointException(); } 使用隱式異常優(yōu)化以后: try{ return foo.value; }catch(segment_fault){ uncommon_trap(); } 當foo極少為空時,隱式異常優(yōu)化是值得的,但是foo經(jīng)常為空,這樣的優(yōu)化反而會讓程序變慢,而HotSpot虛擬機會根據(jù)運行期收集到的Profile信息自動選擇最優(yōu)方案。
方法內聯(lián)
方法內聯(lián)能去除方法調用的成本,同時也為其他優(yōu)化建立了良好的基礎,因此各種編譯器一般會把內聯(lián)優(yōu)化放在優(yōu)化序列的最靠前位置,然而由于Java對象的方法默認都是虛方法,因此方法調用都需要在運行時進行多態(tài)選擇,為了解決虛方法的內聯(lián)問題,首先引入了“類型繼承關系分析(CHA)”的技術。
1.在內聯(lián)時,若是非虛方法,則可以直接內聯(lián) 2.遇到虛方法,首先根據(jù)CHA判斷此方法是否有多個目標版本,若只有一個,可以直接內聯(lián),但是需要預留一個“逃生門”,稱為守護內聯(lián),若在程序的后續(xù)執(zhí)行過程中,加載了導致繼承關系發(fā)生變化的新類,就需要拋棄已經(jīng)編譯的代碼,退回到解釋狀態(tài)執(zhí)行,或者重新編譯。 3.若CHA判斷此方法有多個目標版本,則編譯器會使用“內聯(lián)緩存”,第一次調用緩存記錄下方法接收者的版本信息,并且每次調用都比較版本,若一致則可以一直使用,若不一致則取消內聯(lián),查找虛方法表進行方法分派。
逃逸分析
逃逸分析的基本行為就是分析對象動態(tài)作用域,當一個對象被外部方法所引用,稱為方法逃逸;當被外部線程訪問,稱為線程逃逸。若能證明一個對象不會被外部方法或進程引用,則可以為這個變量進行一些優(yōu)化:
1.棧上分配:如果確定一個對象不會逃逸,則可以讓它分配在棧上,對象所占用的內存空間就可以隨棧幀出棧而銷毀。這樣可以減小垃圾收集系統(tǒng)的壓力。 2.同步消除:線程同步相對耗時,如果確定一個變量不會逃逸出線程,那這個變量的讀寫不會有競爭,則對這個變量實施的同步措施也就可以消除掉。 3.標量替換:如果逃逸分析證明一個對象不會被外部訪問,并且這個對象可以被拆散的話,那么程序真正執(zhí)行的時候可以不創(chuàng)建這個對象,改為直接創(chuàng)建它的成員變量,這樣就可以在棧上分配。
可是目前還不能保證逃逸分析的性能收益必定高于它的消耗,所以這項技術還不是很成熟。
Java虛擬機的即時編譯器與C/C++的靜態(tài)編譯器相比,可能會由于下面的原因導致輸出的本地代碼有一些劣勢: 1.即時編譯器運行占用的是用戶程序的運行時間,具有很大的時間壓力,因此不敢隨便引入大規(guī)模的優(yōu)化技術; 2.Java語言是動態(tài)的類型安全語言,虛擬器需要頻繁的進行動態(tài)檢查,如空指針,上下界范圍,繼承關系等; 3.Java中使用虛方法頻率遠高于C++,則需要進行多態(tài)選擇的頻率遠高于C++; 4.Java是可以動態(tài)擴展的語言,運行時加載新的類可能改變原有的繼承關系,許多全局的優(yōu)化措施只能以激進優(yōu)化的方式來完成; 5.Java語言的對象內存都在堆上分配,垃圾回收的壓力比C++大 然而,Java語言這些性能上的劣勢換取了開發(fā)效率上的優(yōu)勢,并且由于C++編譯器所有優(yōu)化都是在編譯期完成的,以運行期性能監(jiān)控為基礎的優(yōu)化措施都無法進行,這也是Java編譯器獨有的優(yōu)勢。
關于如何進行JVM虛擬機中Java的編譯期優(yōu)化與運行期優(yōu)化就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。