Java基礎語法:運算符、包機制、JavaDoc 自增、自減、一元運算符:++、-- 例子:b = a++; -->先給b賦值,a再自增:b=a; a=a+1; b = ++a; -->a先自增,再給b賦值:a=a+1; b=a; 初識Math類 冪運算:Math.pow(a, b):其中a與b都是 ...
程式計數器
用於標識線程執行到了位元組碼文件(class文件)的哪一行,當執行native方法時,值為undefined,各個線程私有
Java虛擬機棧
每個線程獨有,每個方法執行時會創建一個棧幀,用於存儲局部變數表等方法信息,即方法的執行會伴隨著一個棧幀的入棧出棧。
棧幀的組成
如果線程請求的棧深度大於虛擬機允許的最大深度,將拋出StackoverflowError異常
棧最大深度由棧記憶體大小和局部變數表大小確定,即棧大小固定,局部變數表越大,棧深度越小,棧大小可通過-Xss參數設置,預設為1m
如果棧動態擴展失敗,會拋出OutOfMemoryError異常,不過hotspot虛擬機不支持棧動態擴展,申請時失敗了,會拋出OOM異常
本地方法棧
虛擬機執行native方法時使用,與Java虛擬機棧類似,hotspot將二者合在一起
Java堆
用於存儲Java實例對象,垃圾收集器管理的記憶體區域,也稱為GC堆(Garbage Collected Heap)。垃圾收集就發生在該區域,一般分為eden,servivor1,servivor2,old。
另外在Java8之前還有一個永久代,不過永久代實際上不屬於堆記憶體,屬於虛擬機記憶體,同時受垃圾收集器管理(fullgc時觸發垃圾回收)。Java8之後刪除了永久代,將方法區移動至元空間(Metaspace),使用native記憶體,移除虛擬機,防止OOM異常,元空間大小可以通過 -XX:MaxMetaspaceSize參數控制,fullgc時會對元空間進行垃圾回收;將字元串常量遷移到Java堆中。
Java堆所有線程共用,不過可以通過設置TLAB(ThreadLocalAllocationBuffer)給各線程預留記憶體,提高記憶體分配時的效率。
Java堆可以通過-Xms和-Xmx設置初始大小和最大大小(雖然現在的項目一般將兩個值設為相同以減少GC次數),當動態申請記憶體不足時,會拋出OOM異常。
方法區
用於存儲已經被Java虛擬機載入的類型信息,常量,靜態變數和即時編譯器編譯後的代碼緩存等數據。Java8之前通過永久代實現,Java8之後通過元空間實現,實際為一個邏輯記憶體區域,多線程共用。Java的元空間會在fullgc時進行垃圾回收,主要進行類的卸載時方法區的垃圾回收。當記憶體分配出現不足時,會拋出OOM異常。
運行時常量池
屬於方法區的一部分,用於存儲類的版本,方法,欄位,介面等描述信息,同時接受新的常量,例如String.intern()方法將字元串放入常量池中,實際將char[]存儲到堆中的字元串常量池中,將引用存儲到運行時常量池中。當記憶體不足時,拋出OOM異常。
註意,運行時常量池不等於字元串常量池,上面說了,字元傳常量池在堆(老年代)中。
直接記憶體
直接記憶體不屬於JVM虛機記憶體的一部分,屬於native記憶體,JDK1.4後推出的NIO即可以直接操作Native記憶體,通過在Java堆中創建DirectByteBuffer對象控制Native記憶體。