new對(duì)象的過(guò)程
檢查加載
檢驗(yàn)對(duì)應(yīng)的class是否加載到方法區(qū)中谬俄,常量池中是否有對(duì)應(yīng)的符號(hào)引用贝或。
分配內(nèi)存
給對(duì)象分配對(duì)應(yīng)的內(nèi)存,分配策略主要有兩種:指針碰撞(連續(xù)的內(nèi)存空間城菊,移動(dòng)相應(yīng)的便宜量即可)和空閑列表(存在內(nèi)存碎片囚巴,維護(hù)一個(gè)內(nèi)存空閑列表原在,分配內(nèi)存時(shí)對(duì)應(yīng)去列表中尋找)友扰,采取哪種方式主要取決于GC時(shí)是否進(jìn)行了內(nèi)存碎片的整理彤叉;分配時(shí)并發(fā)安全的考慮主要有兩種:CAS和TLAB,前者使用cas指令保證安全庶柿,后者給每一個(gè)線程分配一定的空間。
內(nèi)存空間的初始化
主要是講分配成功的內(nèi)存全部初始化為零值
設(shè)置
這里主要是設(shè)置對(duì)象頭信息秽浇,對(duì)象頭后面會(huì)有解釋
對(duì)象初始化
調(diào)用對(duì)象的初始化方法浮庐,完成對(duì)象的初始化
對(duì)象的內(nèi)存分布
一個(gè)對(duì)象的大小是8byte的倍數(shù),空余部分主要由對(duì)齊填充填滿
在64位lunix下柬焕,非數(shù)組對(duì)象頭大小96bit
對(duì)象頭中klass pointer 是一個(gè)指向方法區(qū)中類信息的指針审残,長(zhǎng)度32bit(開(kāi)啟指針壓縮后)
mark word 長(zhǎng)度64bit,主要有hash,GC狀態(tài),sync裝態(tài)的數(shù)據(jù)
以下面進(jìn)行舉例:
public class A {
private boolean x = true ;
public void test(){
System.out.println(x);
}
}
public class TestObj {
static A a = new A();
public static void main(String[] args) {
//打印a對(duì)象信息
System.out.println(ClassLayout.parseInstance(a).toPrintable());
}
}
對(duì)象頭這個(gè)變化其實(shí)比較復(fù)雜斑举,這里引用子路大神的一幅圖進(jìn)行解釋
hash :hashcode
線程id:持有該對(duì)象的線程搅轿,上鎖狀態(tài)
age:分帶年齡,由于只有4位富玷,故垃圾回收中璧坟,最大經(jīng)歷15次minor gc就會(huì)進(jìn)入老年代,
biased_lock:對(duì)象是否啟用偏向鎖標(biāo)記赎懦,只占1個(gè)二進(jìn)制位雀鹃。為1時(shí)表示對(duì)象啟用偏向鎖,為0時(shí)表示對(duì)象沒(méi)有偏向鎖励两。
lock:2位的鎖狀態(tài)標(biāo)記位
其組合標(biāo)識(shí)以下?tīng)顟B(tài)(按照biased_lock黎茎,lock)
biased_lock | lock | 狀態(tài) |
---|---|---|
0 | 01 | 無(wú)鎖不可偏向 |
1 | 01 | 偏向鎖,無(wú)鎖可偏向 |
0 | 00 | 輕量級(jí)鎖 |
0 | 10 | 重量級(jí)鎖 |
0 | 11 | GC標(biāo)記 |
對(duì)象訪問(wèn)
主要用兩種方式句柄和直接指針
采用句柄方式訪問(wèn)對(duì)象当悔,會(huì)在堆中維護(hù)一個(gè)句柄池傅瞻,通過(guò)這個(gè)句柄池找到對(duì)應(yīng)的對(duì)象引用,優(yōu)點(diǎn)是在對(duì)象進(jìn)行改變的時(shí)候盲憎,直接修改句柄池的引用即可嗅骄,缺點(diǎn)是增加一層結(jié)構(gòu)
直接指針是直接獲取對(duì)象,簡(jiǎn)單方便焙畔,這也是hotspot默認(rèn)的方式
對(duì)象存活判斷
引用計(jì)數(shù)法
計(jì)算對(duì)象引用的次數(shù)掸读,java不使用,因?yàn)闀?huì)有循環(huán)引用的問(wèn)題
可達(dá)性分析
維護(hù)一個(gè)GC roots宏多,從GC roots 進(jìn)行引用鏈計(jì)算儿惫,如果沒(méi)有被GC roots引用到,則認(rèn)為對(duì)象可以回收
幾種常見(jiàn)的GC roots
1伸但,虛擬棧中引用的對(duì)象
2肾请,方法區(qū)的中靜態(tài)變量常量
3,本地方法棧中引用的對(duì)象
4更胖,JVM的內(nèi)部引用铛铁,如各種異常對(duì)象
5隔显,sync持有的對(duì)象
四種引用類型
1,強(qiáng)引用饵逐,直接new的對(duì)象括眠,存活則不能回收
2,軟引用倍权,SoftReference,發(fā)生oom前掷豺,即使存活,依然可以會(huì)被回收薄声,一般用在緩存
3当船,弱引用,WeakReference,發(fā)生GC,就能被回收
4默辨,虛引用德频,PhantomReference,隨時(shí)可能會(huì)被回收缩幸,日常開(kāi)發(fā)一般很少用到
對(duì)象分配策略
1壹置,優(yōu)先分配在堆的eden區(qū)
2,大對(duì)象直接進(jìn)入old區(qū)
3桌粉,長(zhǎng)期存活的對(duì)象的進(jìn)入old區(qū)(15)
4蒸绩,開(kāi)啟逃逸分析后,如果分析認(rèn)為對(duì)象的生命周期只在方法內(nèi)铃肯,則進(jìn)行棧上分配患亿,減少GC壓力