詳細介紹Java性能的優(yōu)化(一)
Java在九十年代中期出現(xiàn)以后,在贏得贊嘆的同時,也引來了一些批評。贏得的贊嘆主要是Java的跨平臺的操作性,即所謂的”Write Once,Run Anywhere”.但由于Java的性能和運行效率同C相比,仍然有很大的差距,從而引來了很多的批評。
對于服務器端的應用程序,由于不大涉及到界面設計和程序的頻繁重啟,Java的性能問題看似不大明顯,從而一些Java的技術,如JSP,Servlet,EJB等在服務器端編程方面得到了很大的應用,但實際上,Java的性能問題在服務器端依然存在。下面來討論Java的性能和執(zhí)行效率。
一.關于性能的基本知識
1.性能的定義
在我們討論怎樣提高Java的性能之前,我們需要明白“性能“的真正含義。我們一般定義如下五個方面作為評判性能的標準。
1) 運算的性能----哪一個算法的執(zhí)行性能***
2) 內存的分配----程序需要分配多少內存,運行時的效率和性能***。
3) 啟動的時間----程序啟動需要多少時間。
4) 程序的可伸縮性-----程序在用戶負載過重的情況下的表現(xiàn)。
5) 性能的認識------用戶怎樣才能認識到程序的性能。
對于不同的應用程序,對性能的要求也不同。例如,大部分的應用程序在啟動時需要較長的時間,從而對啟動時間的要求有所降低;服務器端的應用程序通常都分配有較大的內存空間,所以對內存的要求也有所降低。但是,這并不是所這兩方面的性能可以被忽略。其次,算法的性能對于那些把商務邏輯運用到事務性操作的應用程序來講非常重要??偟膩碇v,對應用程序的要求將決定對各個性能的優(yōu)先級。
2.怎樣才能提高JAVA的性能
提高JAVA的性能,一般考慮如下的四個主要方面:
(1) 程序設計的方法和模式
一個良好的設計能提高程序的性能,這一點不僅適用于JAVA,也適用也任何的編程語言。因為它充分利用了各種資源,如內存,CPU,高速緩存,對象緩沖池及多線程,從而設計出高性能和可伸縮性強的系統(tǒng)。
當然,為了提高程序的性能而改變原來的設計是比較困難的,但是,程序性能的重要性常常要高于設計上帶來的變化。因此,在編程開始之前就應該有一個好的設計模型和方法。
(2) JAVA布署的環(huán)境。
JAVA布署的環(huán)境就是指用來解釋和執(zhí)行JAVA字節(jié)碼的技術,一般有如下五種。即解釋指令技術(Interpreter Technology),及時編譯的技術(Just In Time Compilier Technology), 適應性優(yōu)化技術(Adaptive Optimization Technology), 動態(tài)優(yōu)化,提前編譯為機器碼的技術(Dynamic Optimization,Ahead Of Time Technology)和編譯為機器碼的技術(Translator Technology).
這些技術一般都通過優(yōu)化線程模型,調整堆和棧的大小來優(yōu)化JAVA的性能。
在考慮提高JAVA的性能時,首先要找到影響JAVA性能的瓶頸(BottleNecks),在確認了設計的合理性后,應該調整JAVA布署的環(huán)境,通過改變一些參數(shù)來提高JAVA應用程序的性能。
(3) JAVA應用程序的實現(xiàn)
當討論應用程序的性能問題時,大多數(shù)的程序員都會考慮程序的代碼,這當然是對的,當更重要的是要找到影響程序性能的瓶頸代碼。為了找到這些瓶頸代碼,我們一般會使用一些輔助的工具,如Jprobe,Optimizit,Vtune以及一些分析的工具如TowerJ Performance等。這些輔助的工具能跟蹤應用程序中執(zhí)行每個函數(shù)或方法所消耗掉的時間,從而改善程序的性能。
(4) 硬件和操作系統(tǒng)
為了提高JAVA應用程序的性能,而采用跟快的CPU和更多的內存,并認為這是提高程序性能的唯一方法,但事實并非如此。實踐經(jīng)驗和事實證明,只有遭到了應用程序性能的瓶頸,從而采取適當?shù)梅椒?,如設計模式,布署的環(huán)境,操作系統(tǒng)的調整,才是最有效的。
3.程序中通常的性能瓶頸。
所有的應用程序都存在性能瓶頸,為了提高應用程序的性能,就要盡可能的減少程序的瓶頸。以下是在JAVA程序中經(jīng)常存在的性能瓶頸。
了解了這些瓶頸后,就可以有針對性的減少這些瓶頸,從而提高JAVA應用程序的性能
4. 提高JAVA程序性能的步驟
為了提高JAVA程序的性能,需要遵循如下的六個步驟。
a) 明確對性能的具體要求
在實施一個項目之前,必須要明確該項目對于程序性能的具體要求,如:這個應用程序要支持5000個并發(fā)的用戶,并且響應時間要在5秒鐘之內。但同時也要明白對于性能的要求不應該同對程序的其他要求沖突。
b) 了解當前程序的性能
你應該了解你的應用程序的性能同項目所要求性能之間的差距。通常的指標是單位時間內的處理數(shù)和響應時間,有時還會比較CPU和內存的利用率。
c) 找到程序的性能瓶頸
為了發(fā)現(xiàn)程序中的性能瓶頸,通常會使用一些分析工具,如:TowerJ Application Performance Analyzer或VTune來察看和分析程序堆棧中各個元素的消耗時間,從而正確的找到并改正引起性能降低的瓶頸代碼,從而提高程序的性能。這些工具還能發(fā)現(xiàn)諸如過多的異常處理,垃圾回收等潛在的問題。
d) 采取適當?shù)拇胧﹣硖岣咝阅?
找到了引起程序性能降低的瓶頸代碼后,我們就可以用前面介紹過的提高性能的四個方面,即設計模式,JAVA代碼的實現(xiàn),布署JAVA的環(huán)境和操作系統(tǒng)來提高應用程序的性能。具體內容將在下面的內容中作詳細說明。
e) 只進行某一方面的修改來提高性能
一次只改變可能引起性能降低的某一方面,然后觀察程序的性能是否有所提高,而不應該一次改變多個方面,因為這樣你將不知道到底哪個方面的改變提高了程序的性能,哪個方面沒有,即不能知道程序瓶頸在哪。
f) 返回到步驟c,繼續(xù)作類似的工作,一直達到要求的性能為止。
二. JAVA布署的環(huán)境和編譯技術
開發(fā)JAVA應用程序時,首先把JAVA的源程序編譯為與平臺無關的字節(jié)碼。這些字節(jié)碼就可以被各種基于JVM的技術所執(zhí)行。這些技術主要分為兩個大類。即基于解釋的技術和基于提前編譯為本地碼的技術。
具體可分為如下的五類:
a) 解釋指令技術
其結構圖和執(zhí)行過程如下:
JAVA的編譯器首先把JAVA源文件編譯為字節(jié)碼。這些字節(jié)碼對于JAVA虛擬機(JVM)來講就是機器的指令碼。然后,JAVA的解釋器不斷的循環(huán)取出字節(jié)碼進行解釋并執(zhí)行。
這樣做的優(yōu)點是可以實現(xiàn)JAVA語言的跨平臺,同時生成的字節(jié)碼也比較緊湊。JAVA的一些優(yōu)點,如安全性,動態(tài)性都得保持;但缺點是省生成的字節(jié)碼沒有經(jīng)過什么優(yōu)化,同全部編譯好的本地碼相比,速度比較慢。
b) 及時編譯技術(Just In Time)
及時編譯技術是為了解決指令解釋技術效率比較低,速度比較慢的情況下提出的。
其主要變化是在JAVA程序執(zhí)行之前,又JIT編譯器把JAVA的字節(jié)碼編譯為機器碼。從而在程序運行時直接執(zhí)行機器碼,而不用對字節(jié)碼進行解釋。同時對代碼也進行了部分的優(yōu)化。
這樣做的優(yōu)點是大大提高了JAVA程序的性能。同時,由于編譯的結果并不在程序運行間保存,因此也節(jié)約了存儲空間了加載程序的時間;缺點是由于JIT編譯器對所有的代碼都想優(yōu)化,因此也浪費了很多的時間。
IBM和SUN公司都提供了相關的JIT產品。
c) 適應性優(yōu)化技術(Adaptive Optimization Technology)
同JIT技術相比,適應性優(yōu)化技術并不對所有的字節(jié)碼進行優(yōu)化。它會跟蹤程序運行的成個過程,從而發(fā)現(xiàn)需要優(yōu)化的代碼,對代碼進行動態(tài)的優(yōu)化。對優(yōu)化的代碼,采取80/20的策略。從理論上講,程序運行的時間越長,代碼就越優(yōu)化。其結構圖如下:
其優(yōu)點是適應性優(yōu)化技術充分利用了程序執(zhí)行時的信息,發(fā)行程序的性能瓶頸,從而提高程序的性能;其缺點是在進行優(yōu)化時可能會選擇不當,發(fā)而降低了程序的性能。
其主要產品又IBM,SUN的HotSpot.
d) 動態(tài)優(yōu)化,提前編譯為機器碼的技術(Dynamic Optimization,Ahead Of Time)
動態(tài)優(yōu)化技術充分利用了JAVA源碼編譯,字節(jié)碼編譯,動態(tài)編譯和靜態(tài)編譯的技術。其輸入時JAVA的原碼或字節(jié)碼,而輸出是經(jīng)過高度優(yōu)化的可執(zhí)行代碼和個來動態(tài)庫的混合(Window中是DLL文件,UNIX中是共享庫.a .so文件)。
其優(yōu)點是能大大提高程序的性能;缺點是破壞了JAVA的可移植性,也對JAVA的安全帶來了一定的隱患。
其主要產品是TowerJ3.0.
接下一篇,詳細介紹Java性能的優(yōu)化(二),希望對你有幫助。