<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    隨筆-199  評論-203  文章-11  trackbacks-0
    內存是稀缺的資源,哪怕內存一塊錢一條!如果在編程中使用不當,再大的內存也會耗光。

        一、認識Java的自動垃圾回收

        垃圾回收是Java語言的一大特性,方便了編程,是以消耗性能為代價的。而垃圾在這里只無用的對象。而C++是需要程序員自己寫析構函數來釋放內存的,麻煩,也有可能忘記而導致內存泄露。

        Java語言對內存的分配管理是通過JVM內部機制決定的。程序員可以不關心其處理。

        二、垃圾回收的原理和意義

        Java虛擬機中有個稱之為垃圾回收器的東西,實際上這個東西也許真正不存在,或者是已經集成到JVM中了,但這無關緊要,我們仍然可以稱為為垃圾回收器。

        垃圾回收器的作用是查找和回收(清理)無用的對象。以便讓JVM更有效的使用內存。

        垃圾回收器的運行時間是不確定的,由JVM決定,在運行時是間歇執行的。雖然可以通過System.gc()來強制回收垃圾,但是這個命令下達后無法保證JVM會立即響應執行,但經驗表明,下達命令后,會在短期內執行你的請求。JVM通常會感到內存緊缺時候去執行垃圾回收操作。

        垃圾回收過于頻繁會導致性能下降,過于稀疏會導致內存緊缺。這個JVM會將其控制到最好,不用程序員擔心。但有些程序在短期會吃掉大量內存,而這些恐怖的對象很快使用結束了,這時候也許有必要強制下達一條垃圾回命令,這是很有必要的,以便有更多可用的物理內存。

        從上面了解到,沒有用的對象就是垃圾。準確的說,當沒有任何線程訪問一個對象時,該對象就符合垃圾回收的條件。

        對于String,存在一個字符串池,這個不屬于本文討論的范圍,字符串池中的垃圾回收,算法和這里所討論的垃圾回收完全是兩碼事。但是不得不說的是,字符串的胡亂拼接,往往導致性能急劇下降,尤其是在龐大的循環語句中,拼接字符串就是在讓程序慢性自殺。這也是很多Java程序員容易犯的毛病。

        字符串既然是池,就是為了緩沖,為了有更高的命中率,因此垃圾回收的頻率也許會比JVM對象垃圾回收器要低很多。

        垃圾回收器僅僅能做的是盡可能保證可用內存的使用效率,讓可用內存得到高效的管理。程序員可以影響垃圾回收的執行,但不能控制。

        三、通過編程影響垃圾回收

        雖然程序員無法控制JVM的垃圾回收機制。但是可以通過編程的手段來影響,影響的方法是,讓對象符合垃圾回收條件。

        分別說來有一下幾種:

        1、將無用對象賦值為null.

        2、重新為引用變量賦值。比如:

     

     Person p = new Person("aaa");
     p = new Person("bbb");

     

        這樣,new Person("aaa")這個對象就是垃圾了——符合垃圾回收條件了。

        3、讓相互聯系的對象稱為“島”對象

     

     Person p1 = new Person("aaa");
     Person p2 = new Person("bbb");
     Person p3 = new Person("ccc");
     p1=p2; p2=p3; p3=p1;
     p1=null; p2=null; p3=null;

     

        在沒有對p1、p2、p3置null之前,它們之間是一種三角戀關系。分別置null,三角戀關系依然存在,但是三個變量不在使用它們了。三個Person對象就組成了一個孤島,最后死在堆上——被垃圾回收掉。

        4、強制的垃圾回收System.gc()

        實際上這里的強制,是程序員的意愿、建議,什么時候執行是JVM的垃圾回收器說了算。

        調用垃圾回收也不一定能保證未使用的對象一定能從內存中刪除。

        唯一能保證的是,當你內存在極少的情況,垃圾回收器在程序拋出OutofMemaryException之前運行一次。
      四、很神秘的finalize()方法

        finalize()方法的確很神秘,是因為你不了解其原理。

        原理:1、finalize()方法是Object中的方法。

        2、finalize()方法會在對象被垃圾回收之前被垃圾回收器調用一次,這是Java語言的一種機制。

        3、finalize()方法在任何對象上最多只會被垃圾回收器調用一次。

        陷阱:1、垃圾回收器無法保證垃圾對象能被回收,因此,finalize()方法也無法保證運行。建議不要重寫finalize()方法,即使重寫,也不要在finalize()方法中寫關鍵的代碼。

        2、finalize()方法中可以把自己傳遞個別的對象,這樣就不是垃圾了,避免了被回收。但是當下次這個對象又符合垃圾回收的時候,finalize()方法不會被調用第二次了,而是直接被清理掉了。

        總結:

        理解了垃圾回收的前提是理解Java運行時的內存堆棧模型。

        理解Java垃圾回收的目的是為了對Java內存管理有個認識,在編程時更有效的使用內存。

        不建議為了垃圾回收,手動編寫大量代碼,這是很愚蠢的做法。可以通過簡單的方式去影響即可。

        本文的討論的垃圾回收排除String對象。String的垃圾回收與String池有很很大關系,目前還沒有研究。但是文中已經提及String使用中容易出現的問題。

    posted on 2009-07-05 21:53 Werther 閱讀(2307) 評論(4)  編輯  收藏 所屬分類: 10.Java

    評論:
    # re: Java的垃圾回收總結 2009-07-05 23:35 | 凡客誠品
    不建議為了垃圾回收,手動編寫大量代碼,這是很愚蠢的做法。可以通過簡單的方式去影響即可  回復  更多評論
      
    # re: Java的垃圾回收總結 [未登錄] 2009-07-06 09:27 | kimi
    對 gc 的解釋不太全面  回復  更多評論
      
    # re: Java的垃圾回收總結 2009-07-10 16:44 | 火星漁者
    除了明知道要產生大量垃圾的情況下用System.gc()回收一下,其余的時候幾乎不用管它。  回復  更多評論
      
    # re: Java的垃圾回收總結 2010-03-23 11:26 | HoldBelief
    我覺得你的三角戀關系是錯誤的, p1=p2; p2=p3; p3=p1; 之后根本沒有形成三角戀關系,
    因為此時 p1 = p3 = new Person("bbb");
    p2 = new Person("ccc");
    這里根本沒有 new Person("aaa")的事情, 你自己運行一遍這個例子就知道了,
    當p1=p2; p2=p3; p3=p1; 這句代碼結束后, new Person("aaa")就已經被GC回收了  回復  更多評論
      
    主站蜘蛛池模板: 一级毛片直播亚洲| 亚洲国产高清视频在线观看| 久久大香香蕉国产免费网站| 亚洲国产中文在线二区三区免| 18禁成年无码免费网站无遮挡| 一级毛片不卡免费看老司机| 精品亚洲aⅴ在线观看| 成人免费福利电影| 精选影视免费在线 | 亚洲日本久久一区二区va| 亚洲七七久久精品中文国产| 18女人水真多免费高清毛片| 亚洲a∨无码一区二区| 亚洲国产国产综合一区首页| 日本高清免费中文字幕不卡| 日韩人妻一区二区三区免费| 精品在线观看免费| 亚洲欧洲日本天天堂在线观看| 啊灬啊灬别停啊灬用力啊免费看| 日本一卡精品视频免费| 美女被免费视频网站| 亚洲成电影在线观看青青| 免费人成在线观看播放国产| 一区二区三区在线免费看| 国产偷国产偷亚洲高清在线| 亚洲成AV人片久久| 中文字幕人成人乱码亚洲电影| 在线视频免费观看高清| 97无码人妻福利免费公开在线视频| 亚洲精品理论电影在线观看| 亚洲精品在线播放| 亚洲av中文无码乱人伦在线播放| 尤物永久免费AV无码网站| 亚洲国产精品免费观看| 人妻免费一区二区三区最新| 国产亚洲视频在线观看| 亚洲一日韩欧美中文字幕在线| 亚洲精品视频在线| 亚洲国产三级在线观看| 亚洲日韩精品无码专区网站| 国产精品深夜福利免费观看|