內存優化
java做的系統給人的印象是什么?占內存!說道這句話就會有N多人站出來為java辯護,并舉出一堆的性能測試報告來證明這一點。其實從理論上來講java做的系統并不比其他語言開發出來的系統更占用內存,那么為什么卻有這么N多理由來證明它確實占內存呢?兩個字,陋習。
(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 Boolean(i==3));
通常這些系統中構造的Boolean實例的個數是相當多的,所以系統中充滿了大量Boolean實例小對象,這是相當消耗內存的。Boolean類實際上只要兩個實例就夠了,一個true的實例,一個false的實例。
Boolean類提供兩了個靜態變量:
public static final Boolean TRUE = new Boolean(true);
public static final Boolean FALSE = new Boolean(false);
需要的時候只要取這兩個變量就可以了,
比如:
ps.setBoolean("isClosed",Boolean.TRUE);
那么象2、3句那樣要根據一個boolean變量來創建一個Boolean怎么辦呢?可以使用Boolean提供的靜態方法:
比如:
ps.setBoolean("isClosed",Boolean.valueOf(isClosed));
ps.setBoolean("isClosed",Boolean.valueOf(i==3));
因為valueOf的內部實現是:return (b ? TRUE : FALSE);
所以可以節省大量內存。相信如果Java規范直接把Boolean的構造函數規定成private,就再也不會出現這種情況了。
(2)別用new Integer。
和Boolean類似,java開發中使用Integer封裝int的場合也非常多,并且通常用int表示的數值通常都非常小。SUN SDK中對Integer的實例化進行了優化,Integer類緩存了-128到127這256個狀態的Integer,如果使用Integer.valueOf(int i),傳入的int范圍正好在此內,就返回靜態實例。這樣如果我們使用Integer.valueOf代替new Integer的話也將大大降低內存的占用。如果您的系統要在不同的SDK(比如IBM SDK)中使用的話,那么可以自己做了工具類封裝一下,比如IntegerUtils.valueOf(),這樣就可以在任何SDK中都可以使用這種特性。
(3)用StringBuffer代替字符串相加。這個我就不多講了,因為已經被人講過N次了。我只想將一個不是笑話的笑話,我在看國內某“著名”java開發的WEB系統的源碼中,竟然發現其中大量的使用字符串相加,一個拼裝SQL語句的方法中竟然最多構造了將近100個string實例。無語中!
(4)過濫使用哈希表,有一定開發經驗的開發人員經常會使用hash表(hash表在JDK中的一個實現就是HashMap)來緩存一些數據,從而提高系統的運行速度。比如使用HashMap緩存一些物料信息、人員信息等基礎資料,這在提高系統速度的同時也加大了系統的內存占用,特別是當緩存的資料比較多的時候。其實我們可以使用操作系統中的緩存的概念來解決這個問題,也就是給被緩存的分配一個一定大小的緩存容器,按照一定的算法淘汰不需要繼續緩存的對象,這樣一方面會因為進行了對象緩存而提高了系統的運行效率,同時由于緩存容器不是無限制擴大,從而也減少了系統的內存占用。現在有很多開源的緩存實現項目,比如ehcache、oscache等,這些項目都實現了FIFO、MRU等常見的緩存算法。
(5)避免過深的類層次結構和過深的方法調用。因為這兩者都是非常占用內存的(特別是方法調用更是堆棧空間的消耗大戶)。
(6)變量只有在用到它的時候才定義和實例化。
(7)盡量避免使用static變量,類內私有常量可以用final來代替。
java內存管理的思想(主要來源于thinking in java)
Java內存管理特點
++:
另外一種常用的內存管理技術是使用計數器,例如COM模型采用計數器方式管理構件,它與有向圖相比,精度行低(很難處理循環引用的問題),但執行效率很高。
★ Java的內存泄露
例子:
List myList=new ArrayList();
for (int i=1;i<100; i++)
{
Object o=new Object();
myList.add(o);
o=null;
}
//此時,所有的Object對象都沒有被釋放,因為變量myList引用這些對象。
★ 對GC操作
★ 內存泄露檢測
★ 軟引用
★ java程序設計中有關內存管理的經驗
1.最基本的建議是盡早釋放無用對象的引用。如:...
A a = new A();
//應用a對象
a = null; //當使用對象a之后主動將其設置為空
….
注:如果a 是方法的返回值,不要做這樣的處理,否則你從該方法中得到的返回值永遠為空,而且這種錯誤不易被發現、排除 2.盡量少用finalize函數。它會加大GC的工作量。
3.如果需要使用經常用到的圖片,可以使用soft應用類型。它盡可能把圖片保存在內存中
4.注意集合數據類型,包括數組、樹、圖、鏈表等數據結構,這些數據結構對GC來說,回收更為復雜。
5.盡量避免在類的默認構造器中創建、初始化大量的對象,防止在調用其自類的構造器時造成不必要的內存資源浪費
6.盡量避免強制系統做垃圾內存的回收,增長系統做垃圾回收的最終時間
7.盡量避免顯式申請數組空間
8.盡量做遠程方法調用類應用開發時使用瞬間值變量,除非遠程調用端需要獲取該瞬間值變量的值。
9.盡量在合適的場景下使用對象池技術以提高系統性能。