自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

JVM 運行時內(nèi)存分代結構

云計算 虛擬化
對于Java應用程序來說,Java堆(Java Heap)是虛擬機所管理的內(nèi)存中最大的一塊。Java堆是被所有線程共享的一塊內(nèi)存區(qū)域,在虛擬機啟動時創(chuàng)建。

對于Java應用程序來說,Java堆(Java Heap)是虛擬機所管理的內(nèi)存中最大的一塊。Java堆是被所有線程共享的一塊內(nèi)存區(qū)域,在虛擬機啟動時創(chuàng)建。此內(nèi)存區(qū)域的唯一目的就是存放對象實例,所有的對象實例都在這里分配內(nèi)存。

Java堆是垃圾收集器管理的內(nèi)存區(qū)域。從回收內(nèi)存的角度看,由于大部分垃圾收集器大部分都是基于分代收集理論設計的,所以 Java 堆中經(jīng)常會出現(xiàn)“新生代”“老年代”“永久代”“Eden空間”“From Survivor空 間”“To Survivor空間”等區(qū)域。這些區(qū)域劃分僅僅是一部分垃圾收集器的共同特性或者說設計風格而已,而非某個Java虛擬機具體實現(xiàn)的固有內(nèi)存布局,不是《Java虛擬機規(guī)范》里對Java堆的官方的定義。比如:Shenandoah、ZGC 就不支持分代。

JDK 1.7 分代結構

在 JDK 1.7 以及之前堆空間分為 3 部分:新生代,老年代,永久代。然后新生代分為:Eden 區(qū), 和兩個 Survivor 區(qū)。如下圖所示

圖片

JDK 1.8 分代結構

在 JDK 1.8 及其以后,堆空間中移除了永久代。為什么刪除永久代的緣由可以閱讀以下文檔:http://openjdk.java.net/jeps/122。其核心原因主要有以下幾點:這是 Hotspot 和 JRockit 虛擬機融合。JRockit 客戶不需要配置永久代(因為JRockit 沒有永久代),習慣不配置永久代。增加元空間解決類加載所需要的內(nèi)存空間,而且元空間默認是自動拓容的。這樣減少內(nèi)存溢出的可能。堆空間移除永久代過后,堆空間的結構如下圖所示:

圖片

運行時數(shù)據(jù)區(qū)結構如下圖所示:

圖片

G1 收集器

G1將新生代,老年代的物理空間劃分取消了。取而代之的是,G1算法將堆劃分為若干個區(qū)域(Region),它仍然屬于分代收集器。不過,這些區(qū)域的一部分包含新生代,新生代的垃圾收集依然采用暫停所有應用線程的方式,將存活對象拷貝到老年代或者Survivor空間。老年代也分成很多區(qū)域,G1收集器通過將對象從一個區(qū)域復制到另外一個區(qū)域,完成了清理工作。這就意味著,在正常的處理過程中,G1完成了堆的壓縮(至少是部分堆的壓縮),這樣也就不會有cms內(nèi)存碎片問題的存在了。

圖片

在G1中,還有一種特殊的區(qū)域,叫Humongous區(qū)域。如果一個對象占用的空間超過了分區(qū)容量50%以上,G1收集器就認為這是一個巨型對象。這些巨型對象,默認直接會被分配在年老代,但是如果它是一個短期存在的巨型對象,就會對垃圾收集器造成負面影響。為了解決這個問題,G1劃分了一個Humongous區(qū),它用來專門存放巨型對象。如果一個H區(qū)裝不下一個巨型對象,那么G1會尋找連續(xù)的H分區(qū)來存儲。為了能找到連續(xù)的H區(qū),有時候不得不啟動Full GC。

對象內(nèi)存分配

對象內(nèi)存分配過程如下:

圖片

下面是具體的幾種內(nèi)存分配規(guī)則描述

對象優(yōu)先分配在 Eden 區(qū)

大多數(shù)情況下,對象在新生代 Eden 區(qū)中分配。當Eden區(qū)沒有足夠空間進行分配時,虛擬機將發(fā)起一次 Minor GC。HotSpot虛擬機提供了-XX:+PrintGCDetails 這個收集器日志參數(shù),告訴虛擬機在發(fā)生垃圾收集行為時打印內(nèi)存回收日志,并且在進程退出的時候輸出當前的內(nèi)存各區(qū)域分配情況。測試代碼:

  1. /** 
  2.  * -XX:+PrintGCDetails 
  3.  */ 
  4. public class GCTest { 
  5.  
  6.     public static void main(String[] args) { 
  7.         byte[] allcation2 = new byte[8000 * 1024]; 
  8.     } 

輸出結果

  1. Heap 
  2.  PSYoungGen      total 38400K, used 11353K [0x0000000795580000, 0x0000000798000000, 0x00000007c0000000) 
  3.   eden space 33280K, 34% used [0x0000000795580000,0x00000007960966f8,0x0000000797600000) 
  4.   from space 5120K, 0% used [0x0000000797b00000,0x0000000797b00000,0x0000000798000000) 
  5.   to   space 5120K, 0% used [0x0000000797600000,0x0000000797600000,0x0000000797b00000) 
  6.  ParOldGen       total 87552K, used 0K [0x0000000740000000, 0x0000000745580000, 0x0000000795580000) 
  7.   object space 87552K, 0% used [0x0000000740000000,0x0000000740000000,0x0000000745580000) 
  8.  Metaspace       used 3017K, capacity 4556K, committed 4864K, reserved 1056768K 
  9.   class space    used 319K, capacity 392K, committed 512K, reserved 1048576K 

我們可以通過內(nèi)存空間的分布可以看出 allcation2 是被分配到 eden 區(qū)中的。

大對象直接進入老年代

大對象就是指需要大量連續(xù)內(nèi)存空間的Java對象(比如:字符串、數(shù)組),JVM 參數(shù) -XX:PretenureSizeThreshold 參數(shù)可以設置大對象的大小,指定大于該設置值的對象直接在老年代分配,不會進入年輕代,這個參數(shù)只有在 Serial 和 ParNew 兩個收集器下有效。比如設置:JVM 參數(shù):-XX:PretenureSizeThreshold=1000000(單位直接)-XX:+UseSerialGC, 在執(zhí)行上面的第一個程序就會發(fā)現(xiàn)大對象直接進入了老年代。這樣做的目的就是避免在Eden區(qū)及兩個Survivor區(qū)之間來回復制,產(chǎn)生大量的內(nèi)存復制操作。

長期存活的對象將進入老年代

HotSpot虛擬機中多數(shù)收集器都采用了分代收集來管理堆內(nèi)存,那內(nèi)存回收時就必須能決策哪些存活對象應當放在新生代,哪些存活對象放在老年代中。為做到這點,虛擬機給每個對象定義了一個對象年齡(Age)計數(shù)器,存儲在對象頭中。對象通常在Eden區(qū)里誕生,如果經(jīng)過第一次 Minor GC 后仍然存活,并且能被 Survivor 容納的話,該對象會被移動到 Survivor 空間中,并且將其對象年齡設為1歲。對象在Survivor區(qū)中每熬過一次Minor GC,年齡就增加1歲,當它的年齡增加到一定程度(默認為15),就會被晉升到老年代中。對象晉升老年代的年齡閾值,可以通過參數(shù) -XX:MaxTenuringThreshold 設置。

動態(tài)對象年齡判斷

為了能更好地適應不同程序的內(nèi)存狀況,HotSpot 虛擬機并不是永遠要求對象的年齡必須達到 -XX:MaxTenuringThreshold 才能晉升老年代,如果在 Survivor 空間中相同年齡所有對象大小的總和大于 Survivor 空間的一半,年齡大于或等于該年齡的對象就可以直接進入老年代,無須等到 -XX:MaxTenuringThreshold 中要求的年齡。

空間分配擔保

在發(fā)生 Minor GC 之前,虛擬機必須先檢查老年代最大可用的連續(xù)空間是否大于新生代所有對象總空間。

本文轉載自微信公眾號「運維開發(fā)故事」

【編輯推薦】

 

責任編輯:姜華 來源: 運維開發(fā)故事
相關推薦

2015-07-20 15:44:46

Swift框架MJExtension反射

2017-04-25 14:39:55

JVM內(nèi)存Java

2009-06-17 15:46:36

Java運行時本機內(nèi)存

2018-10-25 09:04:56

Java虛擬機JVM

2024-03-21 09:15:58

JS運行的JavaScrip

2022-03-21 11:07:43

JVM內(nèi)存字節(jié)碼

2021-05-07 09:40:26

云計算云原生WebAssembly

2019-07-12 09:30:12

DashboardDockerDNS

2021-09-11 15:38:23

容器運行鏡像開放

2020-11-16 09:28:41

函數(shù)內(nèi)存

2018-11-22 12:07:37

Java虛擬機結構

2022-01-17 22:09:50

JVM方法區(qū)數(shù)據(jù)

2021-10-14 09:53:38

鴻蒙HarmonyOS應用

2021-08-18 08:32:09

代碼運行時間示波器

2013-11-26 16:49:55

Android開發(fā)運行時KitKat

2020-12-07 13:31:43

GoMutex開發(fā)者

2023-01-03 09:10:21

2023-07-28 10:42:43

2024-03-20 10:46:00

云原生容器

2022-01-19 08:50:53

設備樹Linux文件系統(tǒng)
點贊
收藏

51CTO技術棧公眾號