中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java虛擬機中的垃圾收集器及對象生存法則是什么

發布時間:2022-02-21 17:13:33 來源:億速云 閱讀:157 作者:iii 欄目:開發技術

這篇文章主要講解了“Java虛擬機中的垃圾收集器及對象生存法則是什么”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“Java虛擬機中的垃圾收集器及對象生存法則是什么”吧!

一、概述

Java堆和方法區這兩個區域有著很顯著的不確定性:

1、一個接口的多個實現類需要的內存可能會不一樣,一個方法所執行的不同條件分支所需要的內存也可能不一樣
2、只有處于運行期間,我們才能知道程序究竟會創建哪些對象,創建多少個對象,這部分內存的分配和回收是動態的

二、對象已死?

1、引用計數法

在對象中添加一個引用計數器,每當有一個地方引用它時,計數器就加一;當引用失效時,計數器就減一;任何時刻計數器為零的對象是不可能再被使用的。

引用計數器雖然占用了一些額外的內存空間來進行計數,原理簡單,判定效率很高;

為什么主流Java虛擬機沒有使用引用計數器來管理內存呢

引用計數法看似簡單的算法有很多例外情況要考慮,必須配合大量額外處理才能保證正確的工作,比如單純的引用計數很難解決對象之間互相循環引用的問題

引用計數器的缺陷

/**
 * @Author: yky
 * @CreateTime: 2020-12-13
 * @Description: 引用計數器的缺陷
 */
public class ReferenceCountingGC {
 public Object instance = null;
 private static final int _1MB = 1024 * 1024;
 /**
  * 這個成員變量唯一作用是占內存
  */
 private byte[] bigSize = new byte[2 * _1MB];
 public static void testGC(){
  ReferenceCountingGC objA = new ReferenceCountingGC();
  ReferenceCountingGC objB = new ReferenceCountingGC();
  objA.instance = objB;
  objB.instance = objA;
  //發生GC,objA、objB能否被回收
  System.gc();
 }
}

運行代碼收查看日志信息發現,這兩個對象均被回收虛擬機并沒有因為這兩個相互引用就放棄回收他們---->Java虛擬機并不是通過計數算法來判斷對象是否存活的;

2、可達性分析算法

該算法的核心思想:通過一系列稱為“GC Roots”的根對象作為起始節點集,從這些結點開始,根據引用關系向下搜索,搜索過程所走過的路徑稱為“引用鏈”,如果某個對象到GC Roots間沒有任何引用鏈相連(圖論話來說從GC Roots到這個對象不可達時,則證明此對象是不可能再被使用的)

對象obj5obj6obj7雖然有關聯,但是他們到GC roots不可達因此他們會被判定為可回收對象

在 Java 語言中,可作為 GC Roots 的對象包括以下幾種

  • 虛擬機棧(棧中的本地變量表)中的引用對象,如各線程被調用的方法堆棧中使用到的參數、局部變量、臨時變量等;

  • 方法區中的類靜態屬性引用的對象,如Java類的引用類型靜態變量;

  • 方法區中的常量引用的對象,如字符串常量里的引用;

  • 本地方法棧總JNI(Navicat方法)引用的對象;

  • Java虛擬機內部的引用

  • 所有被同步鎖(synchronized關鍵字)持有的對象

  • 反應Java虛擬機內部情況的JMXBean、JVMTI中注冊的回調、本地代碼緩存等;

  • 根據用戶所選的垃圾收集器以及當前回收的內存區域不同,還可以有其他對象“臨時性”地加入;

無論通過哪種算法判斷對象是否存活都和“引用”離不開關系。

1)強引用

是指在程序代碼之間普遍存在的引用賦值,Object obj = new Object();這種引用關系。
無論什么情況下,只要強引用關系還在,垃圾收集器就不會回收掉被引用的對象;

2)軟引用

用來描述一些還有用,但非必須的對象。只要軟引用關聯著的對象,在系統將要發生內存溢出前,會把這些對象列進回收范圍之中進行第二次回收;如果這次的回收還沒有足夠的空間,才會拋出內存溢出的異常;

JDK1.2后提供SoftReference類實現軟引用:

Soft reference objects, which are cleared at the discretion of the garbage
collector in response to memory demand. Soft references are most often used
to implement memory-sensitive caches.

3)弱引用

弱引用也被用來描那些非必須對象,強度比軟引用更弱一些,被弱引用關聯的對象只能生存到下一次垃圾收集發生為止,當垃圾收集器開始工作,無論當前內存是足夠,都會回收掉只被弱引用關聯的對象;

JDK1.2后WeakReference類用來實現弱引用:

Weak reference objects, which do not prevent their referents from being

made finalizable, finalized, and then reclaimed. Weak references are most

often used to implement canonicalizing mappings.

4)虛引用

也叫“幽靈引用”、“幻影引用”,最弱的一種引用關系

  • 一個對象是否有虛引用的存在,完全不會對其生存時間構成影響,也無法通過虛引用獲得一個對象的實例;

  • 為一個對象設置虛引用的唯一目的是為了能在這個對象被收集器回收時收到一個系統通知;

PhantomReference類來實現虛引用:

Phantom reference objects, which are enqueued after the collector determines that their referents may otherwise be reclaimed. Phantom references are most often used to schedule post-mortem cleanup actions.

應用需要讀取大量本地圖片

如果每次讀取圖片都從硬盤讀取,則會嚴重影響性能;解決方案:【軟引用或者弱引用】

Map<String,SoftReference<BitMap>> imp = new HashMap<String,SoftReference<BitMap>>

4、生存還是死亡?

在進行過可達性分析后的對象也不一定是非死不可的,該對象進行可達性分析后,發現沒有與GC Roots相連接的引用鏈

  • 這個對象就會第一次被標記起來;對對象是否必要執行finalize()方法進行判斷(已經被虛擬機調用過finalize()方法或者沒有覆蓋finalize()方法都認為是沒有必要執行該finalize()方法)

  • F-Queue隊列中存放該對象,優先級較低的Finalizer線程會去執行它;Gc 會對這個隊列里面的對象再進行一次標記,如果在finalize方法中,對象沒有自己自救的話,它就會被標記回收

  • finalize方法自救自己的辦法是:重新與引用鏈上面的任何一個對象建立連接;如把自己this賦值給某個類或對象的成員變量

結果如下:

finalize method executed !
yes,I am still alive :)
no, i am dead :(

  • 并不鼓勵使用這種辦法來拯救對象,它的運行代價高昂,不確定性大,無法保證順序;

  • finalize方法能做的所有工作,try-finally也可以做的更好,更及時,所以希望忘記這個方法的存在;

5、回收方法區

很多人認為方法區(或者HotSpot虛擬機中的元空間或永久代)是沒有垃圾收集行為的,《Java虛擬機規范》中確實說過可以不要求虛擬機在方法區實現垃圾收集,而且在方法區進行垃圾收集的“性價比”一般比較低:在堆中,尤其是在新生代中,常規應用進行一次垃圾收集一般可以回收70%~95%的空間,而永久代的垃圾收集效率遠低于此。

方法區的垃圾收集主要回收兩部分:廢棄的常量和不再使用的類型;

  • 回收廢棄常量與回收Java堆中的對象非常類似。以常量池中字面量的回收為例:

假如一個字符串“Java”已經進入了常量池中,但是當前系統沒有任何一個字符串對象的值是“Java”,換句話說是沒有任何String對象引用常量池中的“Java”常量,也沒有其他地方引用了這個字面量,如果在這時候發生內存回收,而且必要的話,這個“Java”常量就會被系統清理出常量池。

  • 常量池中的其他類(接口)、方法、字段的符號引用也與此類似。

判定一個常量是否是“廢棄常量”比較簡單。而要判定一個類是否是“無用的類”的條件則相對苛刻許多。類需要同時滿足下面3個條件才能算是“無用的類”:

  1. 該類所有的實例都已經被回收,也就是Java堆中不存在該類的任何實例。

  2. 加載該類的ClassLoader已經被回收。

  3. 該類對應的java.lang.Class對象沒有在任何地方被引用,無法在任何地方通過反射訪問該類的方法。

虛擬機可以對滿足上述3個條件的無用類進行回收,這里說的僅僅是“被允許”,而不是和對象一樣,不使用了就必然會回收。在大量使用反射、動態代理、CGLib等bytecode框架的場景,以及動態生成JSP和OSGi這類頻繁自定義ClassLoader的場景都需要虛擬機具備類卸載的功能,以保證不會被方法區造成過大的內存壓力。

感謝各位的閱讀,以上就是“Java虛擬機中的垃圾收集器及對象生存法則是什么”的內容了,經過本文的學習后,相信大家對Java虛擬機中的垃圾收集器及對象生存法則是什么這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

万州区| 泰安市| 乌拉特后旗| 承德市| 白山市| 剑川县| 响水县| 西贡区| 吴旗县| 江源县| 黑龙江省| 内江市| 文登市| 钟山县| 邯郸县| 宁化县| 剑河县| 雷波县| 庆城县| 海晏县| 治多县| 固原市| 平罗县| 梧州市| 新田县| 长海县| 浮山县| 稻城县| 五华县| 扬中市| 雷波县| 阿鲁科尔沁旗| 肇源县| 西充县| 建昌县| 纳雍县| 壶关县| 洞口县| 江口县| 肃宁县| 昆山市|