一 基本知識
1.1 性能是什麼
在性能調優之前,我們首先來了解一下性能是什麼?關於性能,我想每個學習過Java的人都能列出幾點,甚至可以誇誇其談。在《Java TM Platform Performance》一書中,定義了如下五個方面來作為評判性能的標准:
1) 運算的性能——哪一個算法的執行性能最好?
2) 內存的分配——程序運行時需要耗費多少內存?
3) 啟動的時間——程序啟動需要多長時間?這在Web項目中的影響不大,但要注意部分程序需要部署或運行在客戶端時的情形(比如applet程序)。
4) 程序的可伸縮性——在壓力負載的情況下,程序的性能如何?
5) 性能的感知——用戶在什麼情況下會覺得程序的性能不好?
以上五個方面,在具體的使用場景可以有選擇的去評判。至於這五方面的性能調優,在後續的章節中將會陸續的給以相應的性能調優策略。
1.2 調優的規則
我們只需要關心對我們程序有影響,可以察覺到的性能問題,而不是每一個類中的每一個方法我們都需要想方設法的提高性能。如果程序的性能沒有達到我們所期望的要求,我們才需要考慮如何優化性能。同樣的,晦澀的代碼雖然提高了程序的性能,但同時可能帶給我們的是維護的噩夢。我們需要折中的考慮以上兩種情況,使得程序的代碼是優美的,並且運行的足夠快,達到客戶所期望的性能要求。
優化代碼甚至會導致不良的結果,Donald Knuth(一位比較牛比較有影響的人物,具體是誰,我也忘了,誰知道,可以告訴我一下,謝謝!)曾說過,“Premature optimization is the root of all evil”。在開始性能調優前,需要先指出不優化代碼的一些理由。
1) 如果優化的代碼已經正常工作,優化後可能會引入新的bug;
2) 優化代碼趨向於使代碼更難理解和維護;
3) 在一個平台上優化的代碼,在另一個平台上可能更糟;
4) 花費很多時間在代碼的優化上,提高了很少的性能,卻導致了晦澀的代碼。 確實,在優化前,我們必須認真的考慮是否值得去優化。
1.3 調優的步驟
一般我們提高應用程序的性能劃分為以下幾個步驟:
1) 明確應用程序的性能指標,怎樣才符合期望的性能需求;
2) 在目標平台進行測試;
3) 如果性能已經達到性能指標,Stop;
4) 查找性能瓶頸;
5) 修改性能瓶頸;
6) 返回到第2步。
二 JDK調優
2.1 選擇合適的JDK版本
不同版本的JDK,甚至不同廠家的JDK可能都存在著很大的差異,對於性能優化的程度不同。一般來說,盡可能選擇最新發布的穩定的JDK版本。最新的穩定的JDK版本相對以前的JDK版本都會做一些bug的修改和性能的優化工作。
2.2 垃圾收集Java堆的優化
垃圾收集就是自動釋放不再被程序所使用的對象的過程。當一個對象不再被程序所引用時,它所引用的堆空間可以被回收,以便被後續的新對象所使用。垃圾收集器必須能夠斷定哪些對象是不再被引用的,並且能夠把它們所占據的堆空間釋放出來。如果對象不再被使用,但還有被程序所引用,這時是不能被垃圾收集器所回收的,此時就是所謂的“內存洩漏”。監控應用程序是否發生了內存洩漏,有一個非常優秀的監控工具推薦給大家——Quest公司的JProbe工具,使用它來觀察程序運行期的內存變化,並可產生內存快照,從而分析並定位內存洩漏的確切位置,可以精確定位到源碼內。這個工具的使用我在後續的章節中還會做具體介紹。
Java堆是指在程序運行時分配給對象生存的空間。通過-mx/-Xmx和-ms/-Xms來設置起始堆的大小和最大堆的大小。根據自己JDK的版本和廠家決定使用-mx和-ms或-Xmx和-Xms。Java堆大小決定了垃圾回收的頻度和速度,Java堆越大,垃圾回收的頻度越低,速度越慢。同理,Java堆越小,垃圾回收的頻度越高,速度越快。要想設置比較理想的參數,還是需要了解一些基礎知識的。 Java堆的最大值不能太大,這樣會造成系統內存被頻繁的交換和分頁。所以最大內存必須低於物理內存減去其他應用程序和進程需要的內存。而且堆設置的太大,造成垃圾回收的時間過長,這樣將得不償失,極大的影響程序的性能。以下是一些經常使用的參數設置:
1) 設置-Xms等於-XmX的值;
2) 估計內存中存活對象所占的空間的大小,設置-Xms等於此值,-Xmx四倍於此值;
3) 設置-Xms等於-Xmx的1/2大小;
4) 設置-Xms介於-Xmx的1/10到1/4之間;
5) 使用默認的設置。
大家需要根據自己的運行程序的具體使用場景,來確定最適合自己的參數設置。 除了-Xms和-Xmx兩個最重要的參數外,還有很多可能會用到的參數,這些參數通常強烈的依賴於垃圾收集的算法,所以可能因為JDK的版本和廠家而有所不同。但這些參數一般在web開發中用的比較少,我就不做詳細介紹了。在實際的應用中注意設置-Xms和-Xmx使其盡可能的優化應用程序就行了。對於性能要求很高的程序,就需要自己再多研究研究Java虛擬機和垃圾收集算法的機制了。可以看看曹曉鋼翻譯的《深入Java虛擬機》一書。