Java恼琼,相較于C++等語言,具備自動(dòng)垃圾回收機(jī)制拆讯。如何確定哪些對象是“垃圾”脂男,以及如何回收“垃圾”?
第一种呐,首先確認(rèn)哪些是“垃圾對象”:檢索原理有“引用計(jì)數(shù)法”和“根對象追蹤法”宰翅。引用計(jì)數(shù)法即統(tǒng)計(jì)內(nèi)存對象被其他對象引用的情況,如果不存在引用說明是垃圾對象爽室,但是這種方法會(huì)遺漏“引用孤島”對象汁讼,例如兩個(gè)相互引用的A和B對象(均不再被其他對象引用);根對象追蹤法即從根對象為起點(diǎn),跟蹤標(biāo)記相應(yīng)鏈路上對象嘿架,完成標(biāo)記后瓶珊,沒做標(biāo)記的均可視為垃圾對象。
第二耸彪,垃圾回收基本策略:從屬性來看伞芹,主要包括標(biāo)記復(fù)制、標(biāo)記清除蝉娜、標(biāo)記壓縮三種丑瞧。標(biāo)記復(fù)制即通過兩個(gè)區(qū)域A和B,將A中可用對象復(fù)制到B區(qū)域(可用)蜀肘,清空A區(qū)域绊汹,A和B區(qū)域循環(huán)使用,特點(diǎn)是浪費(fèi)內(nèi)存扮宠,而且當(dāng)復(fù)制對象過多時(shí)造成的STW過長西乖,性能抖動(dòng);標(biāo)記清除即直接清除區(qū)域內(nèi)的垃圾對象坛增,特點(diǎn)是快速获雕、但空間碎片較多,造成內(nèi)存浪費(fèi)嚴(yán)重收捣,性能較差届案;標(biāo)記壓縮即清除區(qū)域內(nèi)垃圾對象,并將可用對象遷移到一起罢艾,形成較大的連續(xù)可用區(qū)域楣颠,但當(dāng)可用對象較大時(shí)STW過程,性能抖動(dòng)嚴(yán)重咐蚯。
第三童漩,結(jié)合JVM內(nèi)存區(qū)域,確定運(yùn)用回收策略:從時(shí)空來看春锋,在垃圾回收時(shí)矫膨,應(yīng)針對JVM內(nèi)存區(qū)域特點(diǎn)分區(qū)和分代考慮。Java垃圾回收涉及到的區(qū)域包括新生代(eden期奔,from/to)侧馅,年老代(tenured),方法區(qū)(在JDK1.8以后呐萌,改為元數(shù)據(jù)區(qū))馁痴,以及直接內(nèi)存區(qū)(特殊情況下也會(huì)進(jìn)行垃圾回收)。新生代90%的對象是瞬間存在的搁胆,新生代生存下來的對象最終會(huì)進(jìn)入老年代弥搞,老年代的對象一般能長久存活邮绿;方法區(qū)一般存儲(chǔ)類元數(shù)據(jù)信息,直接內(nèi)存一般在NIO運(yùn)用等場景時(shí)會(huì)調(diào)用(直接內(nèi)存適用于一次創(chuàng)建攀例,多次使用的場景)船逮。
第四,結(jié)合JVM粤铭,采用何種回收收集器:主要包括四種收集器挖胃,1、串行收集器梆惯,新生代使用標(biāo)記復(fù)制算法酱鸭,年老代使用標(biāo)記壓縮算法;2垛吗、并行收集器,新生代使用標(biāo)記復(fù)制算法凹髓,年老代使用標(biāo)記壓縮算法,包括注重系統(tǒng)停頓時(shí)間和吞吐量兩種指標(biāo)的垃圾回收器怯屉;3蔚舀、CMS垃圾回收器,主要注重系統(tǒng)停頓時(shí)間STW抖動(dòng)問題的回收器锨络;4赌躺、G1垃圾回收器,分區(qū)和分代垃圾回收羡儿。
后面將會(huì)通過兩篇博文著重介紹目前在生產(chǎn)上運(yùn)用比較多的CMS和G1兩種垃圾回收器礼患。