java優(yōu)化手冊_第1頁
java優(yōu)化手冊_第2頁
java優(yōu)化手冊_第3頁
java優(yōu)化手冊_第4頁
java優(yōu)化手冊_第5頁
已閱讀5頁,還剩5頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權說明:本文檔由用戶提供并上傳,收益歸屬內容提供方,若內容存在侵權,請進行舉報或認領

文檔簡介

1、.1. 底層篇1.1. 基本數(shù)據(jù)類型時間分析int的運算速度最快,short次之,byte再次之,long再次之。float和double運算速度最慢。除法比乘法慢的太多,基本上除法是乘法的9倍時間。當然,除了浮點型外。根據(jù)intel cpu的參考數(shù)據(jù),乘法計算時間是移位運算的4-5倍是比較正常的。long類型的計算很慢,建議一般少使用它。double運算速度和float相當;浮點的乘法比除法要慢。但是,這個結果并不能真正說明問題。這個結果只是一個一般性的,在特殊情況下,乘法還是比除法快,比如:floata * floatb 仍然是比floata / (1/floatb)快。從實際的數(shù)據(jù)結果來

2、講,乘法的時候,乘數(shù)越小速度越快,特別是在乘數(shù)比3小的時候,乘法時耗接近20,大于4的時候,幾乎都是600的時耗。除法恰好相反,除數(shù)大于1的時候,時耗一般都是350,可是,當除數(shù)小于1的時候,時耗就變成了700了。對于大家關心的移位和乘除2的問題,jdk5.0已經(jīng)做了部分處理。即“var *=2”和“var =1”耗費一樣。但是,除法并沒有進行這類處理,即“var /= 2”耗費和基本的除法一樣。1.2. 類和接口調用時間分析1.2.1. 類的創(chuàng)建 雖然面向對象思想已經(jīng)深入人心,但他在帶來快捷方面的編程風格的時候,也帶來了低下的效率。在java中,反應最快的是object類(這也是顯然的),建

3、立一個新的object類時耗僅僅為20單位。而一個空類(即沒有聲明任何methods和fields)的建立時間則增加到了驚人的400單位。如果再給類增加一些字段的話,時間耗費并沒有特別大的增加,每增加一個int類型字段大概增加30個單位。僅僅就創(chuàng)建時間來說,內嵌的類型都有不錯的表現(xiàn)。比如,創(chuàng)建一個int數(shù)組(僅僅包含一個元素)的時間只比創(chuàng)建一個object對象的時間多一倍。當然,如果你創(chuàng)建的數(shù)組對象包含1000個元素的話,其創(chuàng)建時間顯然還會加上內存管理的時間了,它的時間大概是1萬個時間單位。請注意,我們這里討論的時間單位其實十分小,1萬個時間單位也僅僅只是有0.006毫秒(0.000006秒)

4、。創(chuàng)建一個byte、short、int、long、float和double數(shù)組對象的時間消耗幾乎是一樣的。1.2.2. 方法的調用java在這個方面有一點做得很好,就是調用一個只有很少量代碼的方法的時耗和直接把這段代碼寫到本地的時耗相差很小。當然不包括需要分配很多本地變量的情況。調用本類(this指針)的方法是最快的,時間在1-2個單位。調用其它類的靜態(tài)方法也很快,速度和調用本來方法差不多。調用其它類的非靜態(tài)方法速度就慢一些,在1.5-2.5個時間單位之間。調用繼承接口的方法是十分緩慢的,是調用普通方法的3倍。但是,如果在實現(xiàn)接口的時候加上final關鍵字的話,調用這個方法的時耗就和普通方法差

5、不多了。最慢的是已經(jīng)同步化了的方法。即加上了synchronized關鍵字的方法。調用它的時耗比普通方法高出了近20倍。如果不是萬不得已,不要把synchronized加到方法上面,實在不行的話,你可以把它加到代碼塊上去,這種加法比直接加到方法上面快一點。注意,因為方法大部分時候都是完成很多事情的,所以,十分注意調用方法的開銷是沒有必要的,因為這個時間和方法執(zhí)行需要的時間比較起來只是毛毛雨。1.3. 基本操作時間耗費2. 通用篇“通用篇”討論的問題適合于大多數(shù)java應用。2.1. 不用new關鍵詞創(chuàng)建類的實例用new關鍵詞創(chuàng)建類的實例時,構造函數(shù)鏈中的所有構造函數(shù)都會被自動調用。但如果一個對

6、象實現(xiàn)了cloneable接口,我們可以調用它的clone()方法。clone()方法不會調用任何類構造函數(shù)。在使用設計模式(design pattern)的場合,如果用factory模式創(chuàng)建對象,則改用clone()方法創(chuàng)建新的對象實例非常簡單。例如,下面是factory模式的一個典型實現(xiàn):public static credit getnewcredit() return new credit();改進后的代碼使用clone()方法,如下所示:private static credit basecredit = new credit();public static credit getne

7、wcredit() return (credit) basecredit.clone();上面的思路對于數(shù)組處理同樣很有用。2.2. 使用非阻塞i/o版本較低的jdk不支持非阻塞i/o api。為避免i/o阻塞,一些應用采用了創(chuàng)建大量線程的辦法(在較好的情況下,會使用一個緩沖池)。這種技術可以在許多必須支持并發(fā)i/o流的應用中見到,如web服務器、報價和拍賣應用等。然而,創(chuàng)建java線程需要相當可觀的開銷。jdk 1.4引入了非阻塞的i/o庫(java.nio)。如果應用要求使用版本較早的jdk,在這里有一個支持非阻塞i/o的軟件包。請參見sun中國網(wǎng)站的調整java的i/o性能。2.3. 慎

8、用異常異常對性能不利。拋出異常首先要創(chuàng)建一個新的對象。throwable接口的構造函數(shù)調用名為fillinstacktrace()的本地(native)方法,fillinstacktrace()方法檢查堆棧,收集調用跟蹤信息。只要有異常被拋出,vm就必須調整調用堆棧,因為在處理過程中創(chuàng)建了一個新的對象。異常只能用于錯誤處理,不應該用來控制程序流程。2.4. 不要重復初始化變量默認情況下,調用類的構造函數(shù)時, java會把變量初始化成確定的值:所有的對象被設置成null,整數(shù)變量(byte、short、int、long)設置成0,float和 double變量設置成0.0,邏輯值設置成false

9、。當一個類從另一個類派生時,這一點尤其應該注意,因為用new關鍵詞創(chuàng)建一個對象時,構造函數(shù)鏈中的所有構造函數(shù)都會被自動調用。2.5. 盡量使用局部變量調用方法時傳遞的參數(shù)以及在調用中創(chuàng)建的臨時變量都保存在棧(stack)中,速度較快。其他變量,如靜態(tài)變量、實例變量等,都在堆(heap)中創(chuàng)建,速度較慢。另外,依賴于具體的編譯器/jvm,局部變量還可能得到進一步優(yōu)化。請參見盡可能使用堆棧變量。2.6. 位移完成乘法和除法考慮下面的代碼:for (val = 0; val 100000; val +=5) alterx = val * 8; myresult = val * 2; 用移位操作替代乘

10、法操作可以極大地提高性能。下面是修改后的代碼:for (val = 0; val 100000; val += 5) alterx = val 3; myresult = val 1; 修改后的代碼不再做乘以8的操作,而是改用等價的左移3位操作,每左移1位相當于乘以2。相應地,右移1位操作相當于除以2。值得一提的是,雖然移位操作速度快,但可能使代碼比較難于理解,所以最好加上一些注釋。2.7. 避免在循環(huán)體中創(chuàng)建對象,即使該對象占用內存空間不大.for(int i=0;i10000;+i) object obj = new object(); system.out.println(obj=+ob

11、j); 應改成object obj = null; for(int i=0;i10000;+i) obj = new object(); system.out.println(obj=+obj); 2.8. 當做數(shù)組拷貝操作時,采用system.arraycopy()方法2.9. 盡量避免在循環(huán)體中調用方法2.10. 盡量避免在循環(huán)體中使用try-catch 塊2.11. 在多重循環(huán)中,如果有可能,盡量將最長的循環(huán)放在最內層,最短的循環(huán)放在最外層,以減少循環(huán)層間的變換次數(shù)2.12. 如果預知長度,就設置arraylist的長度2.13. 避免在類在構造器的初始化其他類2.14. 盡量避免在構造

12、中對靜態(tài)變量做賦值操作2.15. 不要在類的構造器中創(chuàng)建類的實例3. jsp/servlet篇3.1. 采用out 對象中的print方法代替println()方法3.2. 采用適當?shù)闹党跏蓟痮ut 對象緩沖區(qū)的大小3.3. 盡量采用forward()方法重定向新的jsp3.4. 通過init()方法來緩存一些靜態(tài)數(shù)據(jù)以提高應用性能.3.5. servletoutputstream 取代 printwriter4. db篇4.1. 純jdbc最快4.2. 最重要的是盡量減少與數(shù)據(jù)庫通信的次數(shù),多使用批處理4.3. 設置合適的fetch_size和batch_size4.4. 采用連接池技術4.

13、5. 選擇合適的事務隔離層與及時關閉連接對象4.6. preparedstatement防止sql注入4.7. 盡可能地做批處理更新5. 內存篇5.1. 別用new boolean()在很多場景中boolean類型是必須的,比如jdbc中boolean類型的set與get都是通過boolean封裝傳遞的,大部分orm也是用boolean來封裝boolean類型的,比如:ps.setboolean(isclosed,new boolean(true);ps.setboolean(isclosed,new boolean(isclosed);ps.setboolean(isclosed,new b

14、oolean(i=3);通常這些系統(tǒng)中構造的boolean實例的個數(shù)是相當多的,所以系統(tǒng)中充滿了大量boolean實例小對象,這是相當消耗內存的。boolean類實際上只要兩個實例就夠了,一個true的實例,一個false的實例。boolean類提供兩了個靜態(tài)變量:public static final boolean true = new boolean(true);public static final boolean false = new boolean(false);需要的時候只要取這兩個變量就可以了,比如:ps.setboolean(isclosed,boolean.true);那

15、么象2、3句那樣要根據(jù)一個boolean變量來創(chuàng)建一個boolean怎么辦呢?可以使用boolean提供的靜態(tài)方法: boolean.valueof()比如:ps.setboolean(isclosed,boolean.valueof(isclosed);ps.setboolean(isclosed,boolean.valueof(i=3);因為valueof的內部實現(xiàn)是:return (b ? true : false);所以可以節(jié)省大量內存。相信如果java規(guī)范直接把boolean的構造函數(shù)規(guī)定成private,就再也不會出現(xiàn)這種情況了。5.2. 別用new integer 和boolea

16、n類似,java開發(fā)中使用integer封裝int的場合也非常多,并且通常用int表示的數(shù)值通常都非常小。sun sdk中對integer的實例化進行了優(yōu)化,integer類緩存了-128到127這256個狀態(tài)的integer,如果使用integer.valueof(int i),傳入的int范圍正好在此內,就返回靜態(tài)實例。這樣如果我們使用integer.valueof代替new integer的話也將大大降低內存的占用。如果您的系統(tǒng)要在不同的sdk(比如ibm sdk)中使用的話,那么可以自己做了工具類封裝一下,比如integerutils.valueof(),這樣就可以在任何sdk中都可以

17、使用這種特性。5.3. 用stringbuffer代替字符串相加這個我就不多講了,因為已經(jīng)被人講過n次了。我只想將一個不是笑話的笑話,我在看國內某“著名”java開發(fā)的web系統(tǒng)的源碼中,竟然發(fā)現(xiàn)其中大量的使用字符串相加,一個拼裝sql語句的方法中竟然最多構造了將近100個string實例。無語中!5.4. 過濫使用哈希表有一定開發(fā)經(jīng)驗的開發(fā)人員經(jīng)常會使用hash表(hash表在jdk中的一個實現(xiàn)就是hashmap)來緩存一些數(shù)據(jù),從而提高系統(tǒng)的運行速度。比如使用hashmap緩存一些物料信息、人員信息等基礎資料,這在提高系統(tǒng)速度的同時也加大了系統(tǒng)的內存占用,特別是當緩存的資料比較多的時候。其

18、實我們可以使用操作系統(tǒng)中的緩存的概念來解決這個問題,也就是給被緩存的分配一個一定大小的緩存容器,按照一定的算法淘汰不需要繼續(xù)緩存的對象,這樣一方面會因為進行了對象緩存而提高了系統(tǒng)的運行效率,同時由于緩存容器不是無限制擴大,從而也減少了系統(tǒng)的內存占用?,F(xiàn)在有很多開源的緩存實現(xiàn)項目,比如ehcache、oscache等,這些項目都實現(xiàn)了fifo、mru等常見的緩存算法。5.5. 避免過深的類層次結構和過深的方法調用因為這兩者都是非常占用內存的(特別是方法調用更是堆棧空間的消耗大戶)。5.6. 變量只有在用到它的時候才定義和實例化5.7. 盡量避免使用static變量,類內私有常量可以用final來

19、代替6. jvm篇6.1. 內存管理java的內存管理就是對象的分配和釋放問題。在java中,程序員需要通過關鍵字new為每個對象申請內存空間 (基本類型除外),所有的對象都在堆 (heap)中分配空間。對象的釋放是由gc決定和執(zhí)行的。在java中,內存的分配是由程序完成的,而內存的釋放是由gc完成的,這種收支兩條線的方法簡化了程序員的工作。但也加重了jvm的工作。這也是java程序運行速度較慢的原因之一。6.1.1. gc釋放空間方法監(jiān)控每一個對象的運行狀態(tài),包括對象的申請、引用、被引用、賦值等。當該對象不再被引用時,釋放對象。6.1.2. 內存管理結構java使用有向圖的方式進行內存管理,

20、對于程序的每一個時刻,我們都有一個有向圖表示jvm的內存分配情況。將對象考慮為有向圖的頂點,將引用關系考慮為圖的有向邊,有向邊從引用者指向被引對象。另外,每個線程對象可以作為一個圖的起始頂點,例如大多程序從main進程開始執(zhí)行,那么該圖就是以main進程頂點開始的一棵根樹。在這個有向圖中,根頂點可達的對象都是有效對象,gc將不回收這些對象。如果某個對象 (連通子圖)與這個根頂點不可達(注意,該圖為有向圖),那么我們認為這個(這些)對象不再被引用,可以被gc回收。使用有向圖方式管理內存的優(yōu)缺點java使用有向圖的方式進行內存管理,可以消除引用循環(huán)的問題,例如有三個對象,相互引用,只要它們和根進程

21、不可達的,那么gc也是可以回收它們的。這種方式的優(yōu)點是管理內存的精度很高,但是效率較低。另外一種常用的內存管理技術是使用計數(shù)器,例如com模型采用計數(shù)器方式管理構件,它與有向圖相比,精度行低(很難處理循環(huán)引用的問題),但執(zhí)行效率很高。6.1.3. java的內存泄露java雖然由gc來回收內存,但也是存在泄露問題的,只是比c+小一點。c+所有對象的分配和回收都需要由用戶來管理。即需要管理點,也需要管理邊。若存在不可達的點,無法回收分配給那個點的內存,導致內存泄露。存在無用的對象引用,自然也會導致內存泄露。java由gc來管理內存回收,gc將回收不可達的對象占用的內存空間。所以,java需要考慮的內存泄露問題主要是那些被引用但無用的對象即指要管理

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
  • 4. 未經(jīng)權益所有人同意不得將文件中的內容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內容本身不做任何修改或編輯,并不能對任何下載內容負責。
  • 6. 下載文件中如有侵權或不適當內容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

最新文檔

評論

0/150

提交評論