国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

譯文-java垃圾回收機制

Warren / 2931人閱讀

摘要:原文出處垃圾回收機制標(biāo)記清除算法介紹最主要的理論算法之一,在實踐過程中,為了真實情景需要,需要許多調(diào)整。因此不會僅僅標(biāo)記清除,垃圾回收期間,內(nèi)存整理進程同時在工作。不同內(nèi)存區(qū)域的垃圾收集機制不辣么容易理解。

原文出處:java垃圾回收機制

標(biāo)記清除算法介紹最主要的理論算法之一,在實踐過程中,為了真實情景需要,需要許多調(diào)整。舉一個簡單例子,我們檢查JVM需要做的各種事情,以便我們安全地去創(chuàng)建對象。

清除壓縮

當(dāng)清除期間,JVM必須確保區(qū)域被不可達對象填充。這會(終將會)導(dǎo)致內(nèi)存碎片化,同樣會導(dǎo)致磁盤碎片化,由此產(chǎn)生兩個問題:

寫操作因為尋找下一個足夠尺寸的空間變得耗費時間,這個寫操作不再簡單。

當(dāng)創(chuàng)建新對象的時候,JVM分配一個連續(xù)的空間。如果內(nèi)存碎片遍布每一個點,沒有足夠的空間容納新創(chuàng)建的對象,分配就會發(fā)生錯誤。

為了避免上面的問題,JVM會確保碎片化不會失控。因此不會僅僅標(biāo)記清除,垃圾回收期間,"內(nèi)存整理"進程同時在工作。這個進程重新分配所有的可達對象讓他們緊密排列,消除(或者減少)碎片。下面是示意圖:

分代假設(shè)

正如我們之前提到,垃圾收集會引起應(yīng)用徹底停頓。對象越多回收垃圾花費的時間越長,這是顯而易見的。如果我們把使用一小塊內(nèi)存工作變成可能,那又會怎樣呢?為了研究這種可行性,一些專家發(fā)現(xiàn)應(yīng)用的大多數(shù)內(nèi)存分為以下兩種情況:

絕大多數(shù)對象很快就成為無用對象。

很少部分對象講過很長時間存活下來。

上面的觀察結(jié)果歸類于新生代假設(shè)。基于這個假設(shè):VM內(nèi)存空間被劃分為Young代 和Old代,后者有時也叫做Tenured。

眾多算法在提升GC性能上已經(jīng)取得進展,使得擁有這樣一個獨立易清除的內(nèi)存區(qū)域變成現(xiàn)實。

這種方式雖說不上毫無問題。當(dāng)垃圾收集器收集一個分代中的對象的時候,不同分代中的對象彼此相互引用的時,實際上被當(dāng)作"GC roots"。

但是更更要的一點是,分代假設(shè)并不適用于一些應(yīng)用。自此,因為那些“夭折”和“有可能永生”的對象,GC算法做了優(yōu)化,JVM對那些期待更久生命的對象表現(xiàn)得友好。

內(nèi)存空間

讀者應(yīng)該了解下圖中java堆區(qū)里面的內(nèi)存劃分。不同內(nèi)存區(qū)域的垃圾收集機制不辣么容易理解。應(yīng)該注意,不同GC算法實現(xiàn)細節(jié)可能不同,然而它們的理念是一致的。

Eden

Eden區(qū)是對象被新創(chuàng)建的時候分配的內(nèi)存區(qū)域。在這個區(qū)域中,多線程可以同時創(chuàng)建多個對象。在Eden區(qū)里,Eden 被分成一個或多個Thread Local Allocation Buffer (縮寫:TLAB)。在這些緩存里,JVM允許線程在對應(yīng)的緩存中分配絕大多數(shù)的對象,避免昂貴的多線程同步。

當(dāng)TLAB中不能分配空間時(因為空間不足),JVM會移到共享的Eden區(qū)去分配,如果共享Eden空間也不足時,Young代中垃圾回收器去釋放更多的空間。如果在GC之后還沒有足夠的空間以供使用,對象會在Old代中分配。

當(dāng)Eden回收期間,GC把所有的可達對象標(biāo)記為存活。

我們已經(jīng)提前注意到,對象可以跨代關(guān)聯(lián),因此我們必須有一個快捷途徑去檢查其他代的對象引用Eden區(qū)中的對象。

從一開始就記錄所有的分代引用是不可取的, JVM有自己的機制:卡標(biāo)記。事實上,JVM僅僅標(biāo)記Eden里面有可能Old代引用Eden代的對象的“臟”對象的位置,你可以在Nitsan的博客里了解更多的信息。

標(biāo)記階段完成之后,Eden區(qū)下面所有存活的對象被復(fù)制到Survivor下面的一塊區(qū)域中。現(xiàn)在,Eden區(qū)被清空,重新分配新建對象。正如“標(biāo)記-復(fù)制”的名稱一樣:存活的對象被標(biāo)記,然后復(fù)制(不是移動)到Survivor區(qū)。

Survivor區(qū)

緊鄰Eden區(qū)的下一個區(qū)域時兩個叫做from和to的Survivor區(qū)。需要注意的一點,兩塊區(qū)域中的一塊是空的。

Survivor中的空的區(qū)域會保存下一刻Young代中垃圾回收后的對象。Young代所有存活的對象(包括Eden區(qū)和Survior區(qū)中非空的from區(qū)域)被復(fù)制到Survivor區(qū)的"to"區(qū)域。在這之后,“to”區(qū)域存放所有對象,“from”區(qū)域清空。兩者進行調(diào)換(譯者注:即from變成to,to變成from)。

在兩個區(qū)域進行數(shù)次復(fù)制存活對象操作,直到一些對象足夠成熟(“old enough”)。記住這一點,基于分代假設(shè),一些對象在數(shù)次GC之后存活下來,而且在很長一段時間內(nèi)繼續(xù)被引用。

這樣的“tenured”對象會升級到Old代。這種情況發(fā)生的時候,對象不再從Survuvor區(qū)一個區(qū)域移動到另外一個區(qū)域,而是進入Old區(qū),在成為不可達對象之前它們一直存在在old區(qū)中。

為了確定哪些對象“old enough”,需要為old區(qū)提供一種算法,GC記錄幸存對象的詳細信息。每代GC完成之后,那些依然存活的對象年齡進行增長。每當(dāng)年齡超過設(shè)定的閥值之后,對象才會被升遷到old區(qū)。

實際上閥值被JVM動態(tài)設(shè)定,-XX:+MaxTenuringThreshold 除外,它設(shè)置最高限定。XX:+MaxTenuringThreshold=0 表示跳過Survivor區(qū)的兩個區(qū)域之間復(fù)制過程直接進入old區(qū)。jvm默認閥值是GC循環(huán)15次。在Hotspot是最大值。

Survivor區(qū)空間不足以容納Young代所有存活對象的時候升遷操作被提前觸發(fā)。

Old代

Old代的具體實現(xiàn)細節(jié)巨復(fù)雜。Old代通常被那些幾乎不可能被當(dāng)作垃圾的對象占據(jù)。

Old代GC觸發(fā)的次數(shù)比Young代少。因此,Old代中的存活對象,不會發(fā)生標(biāo)記復(fù)制過程。相反,這些對象保持最小碎片化。這種算法建立在不同維度之上。大體上,分為以下幾步:

通過設(shè)置所有GC roots可達的對象標(biāo)記位來標(biāo)記所有可達對象。

刪除所有的不可達對象。

通過復(fù)制對象并緊密排列在Old區(qū)的頂端壓縮Old區(qū)的空間。

正如你看到上面描述的那樣,Old代的GC必須明確處理壓縮錯左避免過多的碎片。

PermGen

JAVA 8之前中被稱作“Permanent Generation”的特殊區(qū)域。這是以前存放例如class的metadata。而,Permgen還存儲String之類的額外數(shù)據(jù)。實際上為JAVA開發(fā)者添加了許多麻煩,因為很難預(yù)測到底需要多少的空間。這些錯誤的預(yù)測結(jié)果表現(xiàn)形式為java.lang.OutOfMemoryError: Permgen space。除非是類似OutOfMemoryError的原因是真的是因為內(nèi)存泄漏,解決這種問題的簡單方法是增加permgen尺寸。下圖中設(shè)置permgen尺寸的最大值為256M:

java -XX:MaxPermSize=256m

Metaspace

正如預(yù)測metadata是一件紛繁復(fù)雜的事情那樣,JAVA 8移除了Permanent區(qū),換作Metaspace。從那時起,絕大多數(shù)復(fù)雜的事情都被移到Java heap區(qū)。

類定義文件,現(xiàn)在都存入叫做“Metaspace”的區(qū)域中。他相當(dāng)于本地內(nèi)存的一塊區(qū)域。理論上,Metaspace尺寸僅僅受限于JAVA進程可獲得本地內(nèi)存大小。將JAVA開發(fā)人員從僅僅在應(yīng)用多增加一個類就造成java.lang.OutOfMemoryError: Permgen space的困境中解脫出來。需要注意的是這個看起來不受限制沒有損失的空間-讓Metaspace無限制的增長你會引起內(nèi)存重交換或者/和本地內(nèi)存分配失敗。

某些場合你希望保護自己,你可以如下圖所示限制Metaspace增長,Metaspace尺寸限制在265M:

java -XX:MaxMetaspaceSize=256m

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/66337.html

相關(guān)文章

  • 譯文】Node.js垃圾回收機制-基礎(chǔ)

    摘要:正好最近在學(xué)習(xí)的各種實現(xiàn)原理,在這里斗膽翻譯一篇垃圾回收機制原文鏈接。自動管理的機制中,通常都會包含垃圾回收機制。二垃圾回收機制的概念垃圾回收,是一種自動管理應(yīng)用程序所占內(nèi)存的機制,簡稱方便起見,本文均采用此簡寫。 最近關(guān)注了一個國外技術(shù)博客RisingStack里面有很多高質(zhì)量,且對新手也很friendly的文章。正好最近在學(xué)習(xí)Node.js的各種實現(xiàn)原理,在這里斗膽翻譯一篇Node...

    haobowd 評論0 收藏0
  • 譯文-垃圾回收器是什么

    摘要:垃圾回收器追蹤所有正在使用的對象,將無用對象標(biāo)記為垃圾。自動化指針內(nèi)存回收自動化的最好方式之一是使用鉤子函數(shù)。它們可能因為多種原因發(fā)生,但是這種垃圾回收器是最主流的一種。 原文出處:What Is Garbage Collection? 一眼就應(yīng)該從名稱看出垃圾回收機制的含義-查找垃圾,然后丟棄。事實正好相反。垃圾回收器追蹤所有正在使用的對象,將無用對象標(biāo)記為垃圾。請留意,我們開始研究...

    alanoddsoff 評論0 收藏0
  • 譯文-G1收集器

    摘要:原文出處設(shè)計的一個重要目標(biāo)是設(shè)置階段的持續(xù)時長和頻率,因為垃圾收集器可預(yù)測,可配置。收集器盡自己最大努力高概率實現(xiàn)目標(biāo)但不是必然,它會是硬實時。因此名稱是收集器。運行不同使用獨立的收集器。 原文出處:G1 – Garbage First G1設(shè)計的一個重要目標(biāo)是設(shè)置stop-the-world階段的持續(xù)時長和頻率,因為垃圾收集器可預(yù)測,可配置。事實上,G1是一款軟實時的收集器,意味著你...

    missonce 評論0 收藏0
  • 譯文-JVM中CMS收集器

    摘要:原文出處這種垃圾收集器的官方名稱是。使用收集器的名稱。事件時長記錄不同的類型回收期間垃圾收集器線程消耗事件調(diào)用操作系統(tǒng)活著等待系統(tǒng)事件消耗時間應(yīng)用停頓的時鐘時間。現(xiàn)在我們看一些一些任務(wù)的時間,垃圾收集器線程等待很長時間。 原文出處:Concurrent Mark and Sweep 這種垃圾收集器的官方名稱是Mostly Concurrent Mark and Sweep Garbag...

    morgan 評論0 收藏0
  • [譯文] JavaScript工作原理:V8引擎內(nèi)部+5條優(yōu)化代碼的竅門

    摘要:本文將會深入分析的引擎的內(nèi)部實現(xiàn)。該引擎使用在谷歌瀏覽器內(nèi)部。同其他現(xiàn)代引擎如或所做的一樣,通過實現(xiàn)即時編譯器在執(zhí)行時將代碼編譯成機器代碼。這可使正常執(zhí)行期間只發(fā)生相當(dāng)短的暫停。 原文 How JavaScript works: inside the V8 engine + 5 tips on how to write optimized code 幾周前我們開始了一個系列博文旨在深入...

    dreamans 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<