GC是一個項目的一個重要的觀察指標坊夫,什么時候gc,gc什么東西禁谦,如何gc都是需要了解的孕豹,以便能更好的support gc異常的問題
1.內(nèi)存的回收算法
一種比較易想的算法是引用計數(shù)算法,什么時候我們認為對象可以被回收了呢辛辨,就是沒有被引用的時候捕捂,最簡單的方式就是加上引用計數(shù),但是當出現(xiàn)相互引用的時候斗搞,也就是a引用b指攒,同時b又引用a,那么通過這種算法是不能回收的僻焚,事實也證明了允悦,這種情況是可以被回收的,也就是說我們所使用的并不是這一種算法虑啤。
java使用的內(nèi)存回收算法是根搜索算法隙弛,簡單來說就是所有的對象都向上追溯其被引用對象,當發(fā)現(xiàn)這條一個對象的引用鏈都沒有根節(jié)點的時候狞山,就可以回收該對象了全闷。那么問題來了,什么樣的對象可以稱之為根對象萍启,有如下四類
棧中的本地變量表引用的對象
方法區(qū)中static的對象
方法區(qū)中常量引用對象
Native方法(本地方法棧中的方法)引用的對象
對于相互依賴而言总珠,雖然引用計數(shù)不為0,但是引用他的對象并不存在于方法棧,或者堆的方法區(qū)姚淆。也就是說引用他的對象都不是獲得對象,所以要一起回收屡律。
2.垃圾收集算法
(1)標記清除算法:這種算法就是將可回收的內(nèi)存部分標記腌逢,并在搜索完之后將他們刪除。算法復雜度簡單超埋,與之相對應(yīng)的問題就是內(nèi)存碎片化搏讶,有可能再被請求一片連續(xù)數(shù)據(jù)的時候,明明空間夠霍殴,卻要在進行一次gc媒惕。
(2)復制算法:這種算法被應(yīng)用于新生代和survivor代中。剛才提到了清楚標記算法的缺點就是內(nèi)存的碎片化来庭,那么復制算法可以解決這個問題妒蔚,他是將每次保留下來的內(nèi)存資源copy一份,變成一片連續(xù)區(qū)域月弛。這種算法應(yīng)用于Eden和survivor代肴盏,首先因為新生代絕大部分都會在新的gc之后回收掉,所以沒必要為copy區(qū)留很大的空間帽衙,一般是8:1:1菜皂,也就是說有90%的內(nèi)存區(qū)域是可用的。當進行g(shù)c的時候厉萝,會講幸存下來的對象內(nèi)存區(qū)域copy到空的10%的survivor區(qū)恍飘,然后清空90%的另外兩區(qū),被清空的survivor區(qū)作為新的10%待copy區(qū)谴垫。不斷循環(huán)往復下去章母。
(3)標記整理算法:老樣子先總結(jié)一下復制算法的缺點,就是會浪費一部分空置的內(nèi)存弹渔,而且還有一個問題胳施,當存活率很高的時候,每次都要移動大量的數(shù)據(jù)肢专,并不合適舞肆。所以需要另外一種解決碎片化的方案就是標記整理法,應(yīng)用于老年帶博杖。當每一次gc的時候還是先標記要刪除的內(nèi)存區(qū)域椿胯,然后將所有保留區(qū)域的內(nèi)存向一端移動,然后清除其他部分的內(nèi)存剃根。
(4)分代算法:也就是剛才例子中說到的eden survivor和old帶哩盲,作為現(xiàn)在java內(nèi)存回收的主要機制。
3.內(nèi)存回收中其他要注意的問題
(1)當一次gc要保留的數(shù)據(jù)大于survivor的大小的時候要使用一種擔保機制,就是讓超出survivor的部分先進入老年代廉油。
(2)一般來講對象要長期存活并大于“年齡值”才能進入老年代惠险,但是如果當survivor代的大于平均年齡的對象所占空間大于50%,那么提前將大于的這部分轉(zhuǎn)移到老年代抒线。
(3)新生代gc(minor gc)在新生帶的gc班巩,很頻繁,“殺傷力很強”嘶炭。老年代gc(major gc/full gc)當出現(xiàn)以下情況時會做full gc:老年代被寫滿抱慌,方法區(qū)被寫滿,調(diào)用system.gc()眨猎。當然擔保的時候發(fā)現(xiàn)老年代滿了也屬于第一種抑进。
4.一些參數(shù)的記錄
(1)-XX:NewSize和-XX:MaxNewSize :用于設(shè)置年輕代的大小,建議設(shè)為整個堆大小的1/3或者1/4,兩個值設(shè)為一樣大睡陪。
(2)-XX:SurvivorRatio :用于設(shè)置Eden和其中一個Survivor的比值寺渗,這個值也比較重要。
(3)-XX:+PrintTenuringDistribution :這個參數(shù)用于顯示每次Minor GC時Survivor區(qū)中各個年齡段的對象的大小宝穗。
(4).-XX:InitialTenuringThreshol和-XX:MaxTenuringThreshold :用于設(shè)置晉升到老年代的對象年齡的最小值和最大值户秤,每個對象在堅持過一次Minor GC之后,年齡就加1逮矛。