本篇文章將重點分析jvm,涉及到的內容包括jvm內存模型,類加載器,GC回收算法,GC回收器,總體偏向於理論。java
本篇文章不適合初學者,適合具備3年以上開發經驗的技術人員,歡迎你們一塊兒交流分享,文章如有不足之處,歡迎讀者朋友們指出,先感謝。算法
下圖爲官網關於jdk,jre和jvm的架構圖,從該架構圖,很容易看出三者之間關係:數據庫
(1)jdk包含jre,而jre又包含jvmbash
(2)jdk主要用於開發環境,jre主要用於發佈環境,固然,發佈環境用jdk也沒問題,僅僅是性能可能會有點影響,jdk與jre關係有點相似程序debug版本和release版本之間關係數據結構
(3)從文件大小來講,jdk比jre大。從圖中能夠看出,jdk比jre多了一層工具包,如經常使用的javac,java命令等多線程
關於jvm類加載器,可歸納爲以下圖:架構
1.爲何要有類加載器?jvm
(1)將字節碼文件加載到運行時數據區。.java源碼經過Javac命令編譯後造成的字節碼文件(.class),經過類加載器加載進入jvm中的。函數
(2)肯定字節碼文件在運行時數據區的惟一性。相同的字節碼文件,經過不一樣的類加載器,就造成不一樣的文件,所以字節碼文件在運行時數據區的惟一性是由字節碼文件和加載它的類加載器共同決定的工具
2.類加載器的種類
從種類上來劃分,類加載器主要劃分爲四大類
(1)啓動類加載器 (根類加載器Bootstrap ClassLoader):該類加載器位於類加載器的最頂層,主要加載jre核心相關jar包,如 /jre/lib/rt.jar
(2)擴展類加載器(Extension ClassLoader):該類加載器位於類加載器層次的第二層,主要加載 jre擴展相關jar包,如/jre/lib/ext/*.jar
(3)應用程序類加載器(Application ClassLoader) App:該類加載器位於類加載器的第三層,主要加載類路徑(classpaht)下的相關jar包
(4)用戶自定義類加載器(User ClassLoader):該類加載器爲用戶自定義類加載器,主要加載用戶指定的路徑下的相關jar包
3.類加載器的機制(雙親委派)
對於字節碼的加載,類加載機制爲雙親委派,什麼叫雙親委派呢?
類加載器獲取字節碼文件後,不是直接加載,而是將該字節碼文件傳遞給其直接父級類加載器,其直接父加載器又繼續傳遞給其直接父加載器的直接父加載器,依次類推到根父加載器,若根父加載器
能加載,則加載,不然交給其直接孩子加載器加載,直接孩子加載器能加載就加載,若不能,依次類推其直接孩子類加載器,若都不能加載,最後才由用戶自定義類加載器加載。
4.jdk 1.8 如何實現類加載器?
以下爲jdk 1.8 類加載器的實現,採用遞歸方式
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) {
// First, check if the class has already been loaded
Class<?> c = findLoadedClass(name);
if (c == null) {
long t0 = System.nanoTime();
try {
if (parent != null) {
c = parent.loadClass(name, false);
} else {
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
// ClassNotFoundException thrown if class not found
// from the non-null parent class loader
}
if (c == null) {
// If still not found, then invoke findClass in order
// to find the class.
long t1 = System.nanoTime();
c = findClass(name);
// this is the defining class loader; record the stats
sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
sun.misc.PerfCounter.getFindClasses().increment();
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
}
複製代碼
5.破壞雙親委派模型
在某些狀況下,因爲受加載範圍限制,父類加載器沒法加載到須要的文件,所以父類加載器須要委託其子類加載器去加載相應的字節碼文件。
如在jdk中定義的數據庫驅動接口Driver,但該接口的實現卻由不一樣的數據庫廠商來實現,這就產生這樣一個問題:由啓動類(Bootstrap ClassLoader)
執行的DriverManager要加載實現了Driver接口的相關實現類,從而實現統一管理,但Bootstrap ClassLoader只能加載jre/lib下的相應文件,不能加載
由各個廠商實現的Dirver接口相關實現類(Dirver實現類是由Application ClassLoader加載),這時就須要Bootstrap ClassLoader委託其子類加載器加載Driver
來實現,從而破壞了雙親委派模型。
java中的類,在jvm中的生命週期,大概分爲五個階段:
1.加載階段:獲取字節碼二進制流,並將靜態存儲結構轉化成方法區的運行時數據結構,且在方法區生成相應的類對象(java.lang.Class對象),做爲該類的數據訪問入口。
2.鏈接階段:該階段包括三個小階段,即驗證,準備和解析三階段
(1)驗證:確保字節碼文件符合虛擬機規範要求,如元數據驗證,文件格式驗證,字節碼驗證和符號驗證等
(2)準備:爲內的靜態表裏分配內存,而且設置jvm默認值,對於非靜態變量,此階段,不需分配內存。
(3)解析:將常量池內的符號引用轉化爲直接引用
3.初始化階段:類對象使用前的一些必要初始化工做
以下引用自一位博友的觀點,我的認爲解釋得很好。
在 Java 代碼中,若是要初始化一個靜態字段,咱們能夠在聲明時直接賦值,也能夠在靜態代碼塊中對其賦值。
除了 final static 修飾的常量,直接賦值操做以及全部靜態代碼塊中的代碼,則會被 Java 編譯器置於同一方法中,並把它命名爲 < clinit > 。初始化的目的是是爲標記爲
常量值的字段賦值,以及執行< clinit > 方法的過程。Java 虛擬機會經過加鎖來確保類的 < clinit > 方法僅被執行一次。
哪些條件會發生類初始化呢?
(1)當虛擬機啓動時,初始化用戶指定的主類(main函數);
(2)當遇到用於新建目標類實例的 new 指令時,初始化 new 指令的目標類;
(3)當遇到調用靜態方法的指令時,初始化該靜態方法所在的類;
(4)子類的初始化會觸發父類的初始化;
(5)若是一個接口定義了 default 方法,那麼直接實現或者間接實現該接口的類的初始化,會觸發該接口的初始化;
(6)使用反射 API 對某個類進行反射調用時,初始化這個類;
(7)當初次調用 MethodHandle 實例時,初始化該 MethodHandle 指向的方法所在的類。
4.使用階段:jvm中使用對象
5.卸載階段:將對象從jvm中卸載(unload),哪些條件會使jvm發生類卸載呢?
(1)加載該類的類加載器被回收
(2)該類的全部實例已經被回收
(3)該類對應的java.lang.Class對象沒有任何地方被引用
1.JVM內存模型是怎樣的?
以下爲JVM內存模型架構圖,因爲在以前的文章中論述過,這裏就再也不一 一論述,主要講解堆區。
在jdk 1.8前,堆區主要分爲新生代、老年代和永久代。jdk 1.8後,去掉了永久代,增長了MetaSpace區。這裏,主要分享jdk 1.8。
根據jdk1.8,堆區邏輯抽象爲三個部分:
(1)新生代:包括Eden區,S0區(也叫from區),S21(也叫TO區)
(2)老年代
(3)Metaspace區
2.新生代和老年代的內存大小是怎樣的?
根據官方建議,新生代佔三分之一(Eden:S0:S1=8:1:1),老年代佔三分之二,所以內存分配圖以下:
3.GC回收是怎樣進行的?
對象先在Eden區運行,當Eden內存用佔用滿時,Eden會進行兩個操做:回收不用的對象和將未回收對象放入s0區,此時s0區和s1區互換名稱,即s0->s1,s1->s0,Eden區通過一次對象回收後,釋放了空間,當Eden下次再滿時,執行相同步驟,依次循環執行,當Eden區回收後,剩下的對象超過s0容量,則將觸發一次Minor GC,此時將未回收的對象放入老年區,依次循環執行,當Eden區觸發Minor GC時,剩餘的對象容量大於old區剩餘容量時,則old區將觸發一次Major GC,此時便會觸發一次Full GC。須要注意的是,通常發生Major GC,基本都都會伴隨一次Full GC回收,Full GC很是損耗性能,在JVM調優時,要注意。
下圖我在生產環境截的一張GC圖,監控工具VisualVM
4.垃圾回收算法有哪些?
(1)標記-清除算法
該算法分爲2個階段,即標記階段和清除階段,首先標記全部要回收的對象,而後回收被標記的對象。該算法效率低,且容易產生內存碎片。
a.效率低:須要遍歷兩次內存,第一次標記,第二次回收被標記對象
b.因爲是非連續內存片斷,容易產生碎片,當對象過大時,容易發生Full GC
下圖爲標記-清除算法 回收前和回收後對比示意圖
(2)標記-複製算法
該算法解決了「標記-清除」算法效率低和大部份內存碎片問題,它將內存分爲大小相等的兩塊,每次只使用其中一塊,當其中一塊須要回收時,只需將該塊區域還存活的對象複製到另外一塊,而後再把該塊內存一次性清理掉,循環往復。
下圖爲標記-複製算法回收前和回收收簡要示意圖
然而,因爲年輕代大部分對象駐留時間都很是短,98%的對象都很快被回收,存活的對象很是少,不須要按照內存1:1來劃分,而是按照8:1:1來劃分,
將2%存活的對象放在s0(from區)便可。
以下爲按照Eden:s0:s1 =8:1:1 劃分示意圖
(3)標記-整理算法
該算法分爲兩階段,即標記和整理,首先標記全部存活對象,將這些對象向一端移動,而後直接清理掉端邊界之外的內存。因爲老年代的對象存活時間比較長,所以適合用該算法。
標記過程仍與「標記-清除」過程一致,但後續步驟不是直接對可回收對象進行清理,而是讓全部存活對象向一端移動,而後直接清理掉端邊界之外的內存。
以下爲"標記-整理算法"回收期和回收後示意圖
(4)分代收集算法
該算法爲目前jvm算法,採用分代思想,模型以下:
5.常見GC回收器有哪些?
(1)SerialGC
SerialGC又叫串行回收器,也是最基礎的GC回收器,主要適用於單核cpu,新生代採用複製算法, 老年代採用標記-壓縮算法,在運行的過程當中須要暫停應用程序,
所以會形成STW問題,在JVM標註參數爲:-XX:+UseSerialGC 。
(2)ParallelGC
ParallelGC基於SerialGC,主要解決SerialGC串行問題,改成並行問題,解決多線程問題,但一樣會產生STW問題,jvm關鍵參數:
a.-XX:+UseParNewGC,表示新生代並行(複製算法) 老年代串行(標記-壓縮)
b.XX:+UseParallelOldGC,老年代也是並行
(3)CMS GC
CMSGC屬於老年代回收器,採用「標記-清除算法」,不會發生STW問題,在jvm中參數設置:
-XX:+UseConcMarkSweepGC,表示老年代使用CMS收集器
(4)Garbage First
Garbage First面向jvm垃圾收集器 ,它知足短期停頓的同時達到一個高的吞吐量,適用於多核cpu和大內存的服務端,也是jdk9的默認垃圾回收器。
深刻分析了JVM內存模型,其中重點分析了jdk,jre和jvm關係,jvm類加載器,jvm堆內存劃分,GC回收器和GC回收算法等,總體偏向於理論,因爲篇幅有限,本篇文章未分析這些技術在JVM實際調優中是如何運用的,將在接下來的文章中與你們分享。