JVM 垃圾回收

来源:https://www.cnblogs.com/luojiahu/archive/2018/01/08/8245488.html
-Advertisement-
Play Games

JVM 垃圾回收 這篇文章嘗試對當前主流的JVM垃圾回收機制進行簡要介紹。 一 垃圾回收討論的範圍 JVM 的記憶體分為方法區,JVM棧,本地方法棧,堆,程式計數器等幾個部分。 其中程式計數器,JVM棧,本地方法棧三部分的生命周期與線程相同,隨著線程的回收這幾部分記憶體其生命周期自然結束而得以回收。 而 ...


JVM 垃圾回收

 

這篇文章嘗試對當前主流的JVM垃圾回收機制進行簡要介紹。

 

一 垃圾回收討論的範圍

JVM 的記憶體分為方法區,JVM棧,本地方法棧,堆,程式計數器等幾個部分。

其中程式計數器,JVM棧,本地方法棧三部分的生命周期與線程相同,隨著線程的回收這幾部分記憶體其生命周期自然結束而得以回收。

而方法區用於存儲靜態變數、常量、類信息等數據,堆更是用於創建對象時在其上劃分記憶體。隨著程式的運行,這些數據何時不再有用,需要被回收便是垃圾回收所關註的問題。

二 垃圾回收的基本策略

垃圾回收所要解決的問題便是將記憶體中無用的對象進行回收,以便釋放出記憶體空間用於後續程式運行分配空間的需要。這裡的問題主要有:

  1. 對象何時會無用
  2. 如何回收更加高效

對象何時會無用?

換句話說便是如何判斷對象不會再被任何程式使用。一種解決方案是所謂“引用計數法”,即當程式中每有一處引用該對象時,引用技術器+1,當每有一處引用失效時,對應-1,當一個對象的“引用計數器”歸零時,便認為其可以被回收。這種方案看似行之有效,但一個潛在的問題便是迴圈引用,假設A對象持有B對象的一個引用,B對象同時持有A對象的一個引用,當程式中任何其他地方不再有指向這兩個對象的引用時,A和B仍舊互相持有,這導致A和B將永遠不能被回收,試想程式中出現多處類似的現象,將極易引發記憶體溢出。

主流的程式語言中,通常採用可達性分析(Reachability Analysis)來判斷對象是否無用。可達性分析是指,從GC Roots開始進行搜索,如果一個對象沒有任何GC Roots能夠到達的話便認為該對象不再有用。對於上面所述的互相引用的情況,顯然是從GCRoots不可達的。在Java中,可能的GC Roots對象包括:

JVM棧中引用的對象

方法區中靜態屬性引用的對象

方法區中常量引用的對象

本地方法棧中引用的對象

如何回收更加高效?

所謂高效的問題,可以理解為如何在對應用程式影響最小的情況下完成最垃圾回收。即垃圾回收應儘可能不影響程式的正常運行,同時能夠對不再有用的對象進行高效的回收以為程式的後續運行提供足夠的空間。

三 幾種垃圾收集器

    1.1. Serial / Serial Old收集器

    顧名思義,這是一款單線程收集器。其線上程回收時,僅有一個線程,同時所有的用戶線程均會被停止,即所謂Stop the world,顯而易見,對於要求具有較高性能的伺服器程式來說,這種情況是難以接受的。而對輕量的客戶端程式來說,如果Stop the world僅僅是毫秒或者數十毫秒級別的,那麼採用這種簡單而高效的垃圾回收器也未嘗不可。

    

 

     1.2. ParNew 收集器

    ParaNew收集器可視為Serial收集器的多線程版本,是一款新生代垃圾收集器。由於Parallel Scavenge收集器不能配合CMS收集器使用,ParNew收集器便成為了配合CMS使用的新生代首選收集器,當使用CMS收集器時,ParNew收集器是預設的新生代收集器。也可通過-XX:+UseParNewGC 強制指定使用該收集器。

    

 

    1.3. Prarallel Scavenge收集器

    此收集器也是一款新生代多線程收集器。與ParNew收集器主要不同的是,該收集器主要關註吞吐量這個參數。

    吞吐量=time-of-user-program/(time-of-user-program + time-of-GC)

    吞吐量越高代表系統對CPU的利用率越高,能夠將更多的時間投入到實際的計算中,這適合後臺需要大量計算而交互較少的系統。

    該收集器提供-XX:MaxGCPauseMillis和-XX:GCTimeRatio兩個參數用於控制吞吐量參數。吞吐量=1/(1+GCTimeRatio)。還提供了-XX:+UseAdaptiveSizePolicy用來開啟自適應調整各記憶體分區大小

    1.4. Parallel Old 收集器

    Parallel Old收集器時Parallel Scavenge收集器的老年代版本,在關註吞吐量有限的應用場景中可以結合Parallel Scavenge使用

    1.5. CMS(Concurrent Mark Sweep)收集器

    該收集器以或多最短的停頓時間為目標,採用標記-清除演算法。回收過程可以分為四步:

      a. 初始標記 initial mark

      標記GC Roots能夠直接關聯到達對象,速度很快,需要Stop the world

      b. 併發標記 concurrent mark

      進行GC Roots Tracing, 可與用戶線程一同進行

      c. 重新標記 remark

      修正併發標記期間用戶程式運行導致標記產生變動的部分,並行運行,需要stop the world

      d. 併發清除 concurrent seep

      與用戶線程一併運行

    整個過程中,1和3需要stop the world,但所占用的時間短,而2和4占用的時間長但可以與用戶線程一同進行,因此整體上對用戶程式產生的停頓影響較小。

    

 

    1.6. G1(Garbage First) 收集器

    G1收集器可以獨立完成新生代和老年代的垃圾回收,除了這種傳統劃分外,其將記憶體區域劃分為若幹個獨立的Region。從整體上看G1採用標記-整理演算法,而對每個Region則採用複製演算法,可以有效避免產生大量記憶體碎片導致頻繁GC。此外,G1還會評估每次回收時各Region回收的價值大小,優先回收價值高的Region。主要分為以下幾步

    1.   初始標記 initial mark
    2.   併發標記 concurrent mark
    3.   最終標記 final mark
    4.   篩選回收 live data counting and evacuation

    其他步驟與CMS類似,而第四步主要評估各Region回收價值的大小和成本,按照用戶期望的代價進行回收。

    

 

 

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 今天線上有個NullPointerException 的異常,我翻了一下代碼,拋異常的竟然是switch語句 我有種不祥的預感,本地做了實驗 結果是 Java的switch如果傳入null值,會拋出 java.lang.NullPointerException 的異常 看下麵的代碼,不要指望會跳到d ...
  • 本文講解瞭如果通過Spring boot來實現前後端的交互,首先演示了傳統的數據交互方式,然後重點講解如何設計一個Restful的API,並通過Spring boot來實現相關的API。 ...
  • 窗外下著小雨,作為單身程式員的我逛著逛著發現一篇好東西,來自知乎 你都用 Python 來做什麼?的第一個高亮答案。 到上面去看了看,地址都是明文的,得,趕緊開始吧。 下載流式文件,requests庫中請求的stream設為True就可以啦,文檔在此。 先找一個視頻地址試驗一下: 遭遇當頭一棒: 這 ...
  • 轉到Java以後發現一個很妖的事情,為啥有些方法後邊有個 throws XXXXException 比如下麵的代碼 這個是.NET沒有的一個語法,受查異常 這是摘自《Java核心技術》的解釋,我感覺很明確了 Java語言規範將派生於Error類和RuntimeException類的所有異常稱為非受查 ...
  • 前言 在前面的MyBatis部分內容中,我們已經可以獨立的基於MyBatis構建一個資料庫訪問層應用,但是在實際的項目開發中,我們的程式不會這麼簡單,層次也更加複雜,除了這裡說到的持久層,還有業務邏輯層、視圖層等等,隨著代碼量和需求複雜度逐漸增長,對象的創建、管理以及層與層之間的耦合度等等問題隨之而 ...
  • 在讀這一章時,運算符的內容比較多,不要去死記。現在記不住也沒有關係,現在只要有這個印象。在後面的學習中,會慢慢加深理解,在理解中去記就容易得多了 1、變數 什麼是變數?通俗的講,就是存儲在記憶體中可以變化的量。聯想下當年學的代數: 設正方形的邊長為a,則正方形的面積為a x a。把邊長a看做一個變數, ...
  • 一、while 語句格式 二、do whle 語句格式 三、for 結構 ...
  • 外部模板 傳統 C++ 中,模板只有在使用時才會被編譯器實例化。換句話說,只要在每個編譯單元(文件) 中編譯的代碼中遇到了被完整定義的模板,都會實例化。這就產生了重覆實例化而導致的編譯時間的增加。並且,我們沒有辦法通知編譯器不要出發模板實例化。 C++11 引入了外部模板,擴充了原來的強制編譯器在特... ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...