35 個 Java 代碼性能優化總結 1-10


代碼優化的目標是:

1、減小代碼的體積

2、提高代碼運行的效率


-------------------------------------------------------------------------------------------

代碼優化細節

1、儘量指定類、方法的final修飾符

帶有final修飾符的類是不可派生的。在Java核心API中,有許多應用final的例子,例如java.lang.String,整個類都是 final的。爲類指定final修飾符可以讓類不可以被繼承,爲方法指定final修飾符可以讓方法不可以被重寫。如果指定了一個類爲final,則該 類所有的方法都是final的。Java編譯器會尋找機會內聯所有的final方法,內聯對於提升Java運行效率作用重大,具體參見Java運行期優 化。此舉能夠使性能平均提高50%。


2、儘量重用對象

特別是String對象的使用,出現字符串連接時應該使用StringBuilder/StringBuffer代替。由於Java虛擬機不僅要花時間生成對象,以後可能還需要花時間對這些對象進行垃圾回收和處理,因此,生成過多的對象將會給程序的性能帶來很大的影響。


3、儘可能使用局部變量

調用方法時傳遞的參數以及在調用中創建的臨時變量都保存在棧中速度較快,其他變量,如靜態變量、實例變量等,都在堆中創建,速度較慢。另外,棧中創建的變量,隨着方法的運行結束,這些內容就沒了,不需要額外的垃圾回收。


4、及時關閉流

Java編程過程中,進行數據庫連接、I/O流操作時務必小心,在使用完畢後,及時關閉以釋放資源。因爲對這些大對象的操作會造成系統大的開銷,稍有不慎,將會導致嚴重的後果。


5、儘量減少對變量的重複計算

明確一個概念,對方法的調用,即使方法中只有一句語句,也是有消耗的,包括創建棧幀、調用方法時保護現場、調用方法完畢時恢復現場等。所以例如下面的操作:

for (int i = 0; i < list.size(); i++) 
 
{...} 
 
建議替換爲: 
 
for (int i = 0, int length = list.size(); i < length; i++) 
 
{...}

這樣,在list.size()很大的時候,就減少了很多的消耗


6、儘量採用懶加載的策略,即在需要的時候才創建

例如:

String str = "aaa";if (i == 1) 
{ 
 
list.add(str); 
 
}

建議替換爲:

if (i == 1) 
{ 
String str = "aaa"; 
list.add(str); 
}


7、慎用異常

異常對性能不利。拋出異常首先要創建一個新的對象,Throwable接口的構造函數調用名爲fillInStackTrace()的本地同步方 法,fillInStackTrace()方法檢查堆棧,收集調用跟蹤信息。只要有異常被拋出,Java虛擬機就必須調整調用堆棧,因爲在處理過程中創建 了一個新的對象。異常只能用於錯誤處理,不應該用來控制程序流程。


8、不要在循環中使用try…catch…,應該把其放在最外層

除非不得已。如果毫無理由地這麼寫了,只要你的領導資深一點、有強迫症一點,八成就要罵你爲什麼寫出這種垃圾代碼來了


9、如果能估計到待添加的內容長度,爲底層以數組方式實現的集合、工具類指定初始長度

比如ArrayList、LinkedLlist、StringBuilder、StringBuffer、HashMap、HashSet等等,以StringBuilder爲例:

(1)StringBuilder()// 默認分配16個字符的空間 
(2)StringBuilder(int size)// 默認分配size個字符的空間 
(3)StringBuilder(String str) // 默認分配16個字符+str.length()個字符空間

可以通過類(這裏指的不僅僅是上面的StringBuilder)的來設定它的初始化容量,這樣可以明顯地提升性能。比如 StringBuilder吧,length表示當前的StringBuilder能保持的字符數量。因爲當StringBuilder達到最大容量的時 候,它會將自身容量增加到當前的2倍再加2,無論何時只要StringBuilder達到它的最大容量,它就不得不創建一個新的字符數組然後將舊的字符數 組內容拷貝到新字符數組中—-這是十分耗費性能的一個操作。試想,如果能預估到字符數組中大概要存放5000個字符而不指定長度,最接近5000的2次冪 是4096,每次擴容加的2不管,那麼:

(1)在4096 的基礎上,再申請8194個大小的字符數組,加起來相當於一次申請了12290個大小的字符數組,如果一開始能指定5000個大小的字符數組,就節省了一倍以上的空間

(2)把原來的4096個字符拷貝到新的的字符數組中去

這樣,既浪費內存空間又降低代碼運行效率。所以,給底層以數組實現的集合、工具類設置一個合理的初始化容量是錯不了的,這會帶來立竿見影的效果。但 是,注意,像HashMap這種是以數組+鏈表實現的集合,別把初始大小和你估計的大小設置得一樣,因爲一個table上只連接一個對象的可能性幾乎爲 0。初始大小建議設置爲2的N次冪,如果能估計到有2000個元素,設置成new HashMap(128)、new HashMap(256)都可以。



10、當複製大量數據時,使用System.arraycopy()命令

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章