棧
本地方法棧
堆
方法區(qū)

JVM
- 如果線程正在執(zhí)行一個java方法,則程序計數(shù)器記錄的是正在執(zhí)行的字節(jié)碼指令的地址;如果執(zhí)行的是native方法,則計數(shù)器值為空Undefined;
- 一個class文件的空間大部分被其常量池占用,而字節(jié)碼只占用小部分,而多個class的常量池總是會有部分重疊, 這樣造成的冗余并不適合傳輸;
- 程序計數(shù)器和棧都是一個線程私有,即一個線程一個棧一個程序計數(shù)器。cpu(或一個內(nèi)核)一次只能執(zhí)行一個線程,因此棧中存放的是一個個依次執(zhí)行的方法,即棧幀。棧幀是線程棧的基本單位;
- 棧內(nèi)的數(shù)據(jù)可以被共享,比如不同的變量值相等會指向同一個地址,這樣可以節(jié)省空間;
- 棧的大小由-Xss控制;
- 棧幀(stack frame)保存了局部變量表、操作數(shù)棧、動態(tài)鏈接、方法出口等信息,一個方法的執(zhí)行就是一個棧幀在棧中入棧到出棧的過程;
- StackOverflowError表示線程申請的棧深度超過jvm允許的深度時的異常;而OutOfMemoryError則是線程無法向可動態(tài)擴(kuò)展的棧申請到足夠內(nèi)存(擴(kuò)展都不夠用);
- HotSpot jvm的本地方法棧和虛擬機(jī)棧是同一個;
- 本地方法棧的用于執(zhí)行本地方法(JNI),本地方法由底層實現(xiàn)使用C語言編寫
(待補充)
。
- 所有對象實例和數(shù)組都要在堆(HEAP)上分配,這句話已經(jīng)過時了(棧上分配、逃逸分析、標(biāo)量替換?)。堆可以在物理上由不連續(xù)的內(nèi)存空間構(gòu)成,目前主流的虛擬機(jī)都是可擴(kuò)展(由-Xms和-Xmx兩個參數(shù)來控制
(待補充)
)的;
- HotSpot jvm的GC不僅僅回收堆區(qū)的對象,還擴(kuò)展到了方法區(qū)(把方法區(qū)作為永久代)。jvm規(guī)范把方法區(qū)描述為堆的一個邏輯部分。但是又不同于堆;
- 方法區(qū)和堆一樣是各個線程共享的區(qū)域。
- 方法區(qū)的作用時保存所有的類class(包括類編譯后的字節(jié)碼)、靜態(tài)static變量、常量,這里的類指的是虛擬機(jī)加載的類(不同于堆上的是類的實例),方法區(qū)由-XX:PermSize和-XX:MaxPermSize控制;
- 類、常量、靜態(tài)變量太多的話真的有可能撐爆方法區(qū)(永久代);
- 類加載后,其常量池(Constant Pool Table)的內(nèi)容將會存放到方法區(qū)的運行時常量池中(Runtime Constant Pool),產(chǎn)量池的特點是動態(tài)性,運行期間可以將新的常量放入池之中;
- 直接內(nèi)存不屬于jvm的內(nèi)存區(qū)域范疇,這里提到主要是因為NIO類可以直接由Native函數(shù)分配堆外內(nèi)存使用,這塊受本機(jī)內(nèi)存限制;
- 對象的創(chuàng)建:
- 檢查是否有對應(yīng)的類放在常量池中,若不存在必需先加在該類;
- 分配內(nèi)存(涉及到內(nèi)存分配方式,指針碰撞或者空閑列表等);
- 將分配的內(nèi)存空間都初始化為零值;
- 對對象進(jìn)行必要的設(shè)置;
- 執(zhí)行<init>方法進(jìn)行初始化;
對象分配內(nèi)存不僅僅要考慮如何劃分空間,還要考慮并發(fā)情況:對象的創(chuàng)建在虛擬機(jī)中是十分頻繁的,假如正在給對象A分配內(nèi)存,指針還沒來得及修改,對象B又要使用原來的指針來分配內(nèi)存
String.intern()是一個Native方法,作用是:
若字符串常量池已經(jīng)包含了一個等于此String對象的字符串,則返回池中等于這個字符串的對象;
否則,將此對象包含的字符串放入常量池中并返回此對象的引用。
- java內(nèi)存模型要求lock、unlock、read、load、assign、use、store、write這八個操作都具有原子性。
- 但是對于64位的數(shù)據(jù)類型long和double,在模型中規(guī)定了沒有被volatile修飾的64位數(shù)據(jù)讀寫操作可以劃分為兩次32位的操作。
- 也就是說允許虛擬機(jī)不保證64位數(shù)據(jù)的load、store、read、write這四個操作的原子性,以上成為long和double的非原子性協(xié)定(Nonatomic Treatment of double and long Variables)。
- 多線程操作64位的未被volatile修飾的long和double時,同時進(jìn)行讀取和修改可能出現(xiàn)讀到一個既不是原始值也不是修改后的值的值。