轉自:https://www.cnblogs.com/ityouknow/p/6482464.htmljvm系列(八):jvm知識點總覽html
在江湖中要練就絕世武功必須內外兼備,精妙的招式和深厚的內功,武功的基礎是內功。對於武功低(就像江南七怪)的人,招式更重要,由於他們不能靠內功直接去傷人,只能靠招式,利刃上優點來取勝了,可是練到高手以後,內功就更主要了。一個內功低的人招式在奇妙也打不過一個內功高的人。好比,你劍法再厲害,一劍刺過來,別人一掌打斷你的劍,你還怎麼使劍法,你一掌打到一個武功高的人身上,那人沒什麼事,卻把你震傷了,你還怎麼打。一樣二者也是相輔相成的,內功深厚以後,原來普通的一招一式威力也會倍增。java
對於搞開發的咱們其實也是同樣,如今流行的框架愈來愈多,封裝的也愈來愈完善,各類框架能夠搞定一切,幾乎不用關注底層的實現,初級程序員只要熟悉基本的使用方法,即可以快速的開發上線;但對於高級程序員來說,內功的修煉卻愈加的重要,好比算法、設計模式、底層原理等,只有把這些基礎熟練以後,才能在開發過程當中知其然知其因此然,出現問題時能快速定位到問題的本質。程序員
對於Java程序員來說,spring全家桶幾乎能夠搞定一切,spring全家桶即是精妙的招式,jvm就是內功心法很重要的一塊,線上出現性能問題,jvm調優更是不可迴避的問題。所以JVM基礎知識對於高級程序員的重要性沒必要言語,我司在面試高級開發的時候,jvm相關知識也一定是考覈的標準之一。本篇文章會根據以前寫的jvm系列文章梳理出jvm須要關注的全部考察點。面試
jvm 整體梳理
jvm體系整體分四大塊:算法
- 類的加載機制
- jvm內存結構
- GC算法 垃圾回收
- GC分析 命令調優
固然這些知識點在以前的文章中都有詳細的介紹,這裏只作主幹的梳理spring
這裏畫了一個思惟導圖,將全部的知識點進行了陳列,由於圖比較大能夠點擊右鍵下載了放大查看。設計模式
類的加載機制
主要關注點:瀏覽器
- 什麼是類的加載
- 類的生命週期
- 類加載器
- 雙親委派模型
什麼是類的加載緩存
類的加載指的是將類的.class文件中的二進制數據讀入到內存中,將其放在運行時數據區的方法區內,而後在堆區建立一個java.lang.Class對象,用來封裝類在方法區內的數據結構。類的加載的最終產品是位於堆區中的Class對象,Class對象封裝了類在方法區內的數據結構,而且向Java程序員提供了訪問方法區內的數據結構的接口。服務器
類的生命週期
類的生命週期包括這幾個部分,加載、鏈接、初始化、使用和卸載,其中前三部是類的加載的過程,以下圖;
- 加載,查找並加載類的二進制數據,在Java堆中也建立一個java.lang.Class類的對象
- 鏈接,鏈接又包含三塊內容:驗證、準備、初始化。1)驗證,文件格式、元數據、字節碼、符號引用驗證;2)準備,爲類的靜態變量分配內存,並將其初始化爲默認值;3)解析,把類中的符號引用轉換爲直接引用
- 初始化,爲類的靜態變量賦予正確的初始值
- 使用,new出對象程序中使用
- 卸載,執行垃圾回收
幾個小問題?
一、JVM初始化步驟 ? 二、類初始化時機 ?三、哪幾種狀況下,Java虛擬機將結束生命週期?
答案參考這篇文章jvm系列(一):java類的加載機制
類加載器
- 啓動類加載器:Bootstrap ClassLoader,負責加載存放在JDK\jre\lib(JDK表明JDK的安裝目錄,下同)下,或被-Xbootclasspath參數指定的路徑中的,而且能被虛擬機識別的類庫
- 擴展類加載器:Extension ClassLoader,該加載器由sun.misc.Launcher$ExtClassLoader實現,它負責加載DK\jre\lib\ext目錄中,或者由java.ext.dirs系統變量指定的路徑中的全部類庫(如javax.*開頭的類),開發者能夠直接使用擴展類加載器。
- 應用程序類加載器:Application ClassLoader,該類加載器由sun.misc.Launcher$AppClassLoader來實現,它負責加載用戶類路徑(ClassPath)所指定的類,開發者能夠直接使用該類加載器
類加載機制
- 全盤負責,當一個類加載器負責加載某個Class時,該Class所依賴的和引用的其餘Class也將由該類加載器負責載入,除非顯示使用另一個類加載器來載入
- 父類委託,先讓父類加載器試圖加載該類,只有在父類加載器沒法加載該類時才嘗試從本身的類路徑中加載該類
- 緩存機制,緩存機制將會保證全部加載過的Class都會被緩存,當程序中須要使用某個Class時,類加載器先從緩存區尋找該Class,只有緩存區不存在,系統纔會讀取該類對應的二進制數據,並將其轉換成Class對象,存入緩存區。這就是爲何修改了Class後,必須重啓JVM,程序的修改纔會生效
jvm內存結構
主要關注點:
jvm內存結構
方法區和對是全部線程共享的內存區域;而java棧、本地方法棧和程序員計數器是運行是線程私有的內存區域。
- Java堆(Heap),是Java虛擬機所管理的內存中最大的一塊。Java堆是被全部線程共享的一塊內存區域,在虛擬機啓動時建立。此內存區域的惟一目的就是存放對象實例,幾乎全部的對象實例都在這裏分配內存。
- 方法區(Method Area),方法區(Method Area)與Java堆同樣,是各個線程共享的內存區域,它用於存儲已被虛擬機加載的類信息、常量、靜態變量、即時編譯器編譯後的代碼等數據。
- 程序計數器(Program Counter Register),程序計數器(Program Counter Register)是一塊較小的內存空間,它的做用能夠看作是當前線程所執行的字節碼的行號指示器。
- JVM棧(JVM Stacks),與程序計數器同樣,Java虛擬機棧(Java Virtual Machine Stacks)也是線程私有的,它的生命週期與線程相同。虛擬機棧描述的是Java方法執行的內存模型:每一個方法被執行的時候都會同時建立一個棧幀(Stack Frame)用於存儲局部變量表、操做棧、動態連接、方法出口等信息。每個方法被調用直至執行完成的過程,就對應着一個棧幀在虛擬機棧中從入棧到出棧的過程。
- 本地方法棧(Native Method Stacks),本地方法棧(Native Method Stacks)與虛擬機棧所發揮的做用是很是類似的,其區別不過是虛擬機棧爲虛擬機執行Java方法(也就是字節碼)服務,而本地方法棧則是爲虛擬機使用到的Native方法服務。
對象分配規則
- 對象優先分配在Eden區,若是Eden區沒有足夠的空間時,虛擬機執行一次Minor GC。
- 大對象直接進入老年代(大對象是指須要大量連續內存空間的對象)。這樣作的目的是避免在Eden區和兩個Survivor區之間發生大量的內存拷貝(新生代採用複製算法收集內存)。
- 長期存活的對象進入老年代。虛擬機爲每一個對象定義了一個年齡計數器,若是對象通過了1次Minor GC那麼對象會進入Survivor區,以後每通過一次Minor GC那麼對象的年齡加1,知道達到閥值對象進入老年區。
- 動態判斷對象的年齡。若是Survivor區中相同年齡的全部對象大小的總和大於Survivor空間的一半,年齡大於或等於該年齡的對象能夠直接進入老年代。
- 空間分配擔保。每次進行Minor GC時,JVM會計算Survivor區移至老年區的對象的平均大小,若是這個值大於老年區的剩餘值大小則進行一次Full GC,若是小於檢查HandlePromotionFailure設置,若是true則只進行Monitor GC,若是false則進行Full GC。
如何經過參數來控制個各個內存區域
參考此文章:jvm系列(二):JVM內存結構
GC算法 垃圾回收
主要關注點:
對象存活判斷
判斷對象是否存活通常有兩種方式:
- 引用計數:每一個對象有一個引用計數屬性,新增一個引用時計數加1,引用釋放時計數減1,計數爲0時能夠回收。此方法簡單,沒法解決對象相互循環引用的問題。
- 可達性分析(Reachability Analysis):從GC Roots開始向下搜索,搜索所走過的路徑稱爲引用鏈。當一個對象到GC Roots沒有任何引用鏈相連時,則證實此對象是不可用的,不可達對象。
GC算法
GC最基礎的算法有三種:標記 -清除算法、複製算法、標記-壓縮算法,咱們經常使用的垃圾回收器通常都採用分代收集算法。
- 標記 -清除算法,「標記-清除」(Mark-Sweep)算法,如它的名字同樣,算法分爲「標記」和「清除」兩個階段:首先標記出全部須要回收的對象,在標記完成後統一回收掉全部被標記的對象。
- 複製算法,「複製」(Copying)的收集算法,它將可用內存按容量劃分爲大小相等的兩塊,每次只使用其中的一塊。當這一塊的內存用完了,就將還存活着的對象複製到另一塊上面,而後再把已使用過的內存空間一次清理掉。
- 標記-壓縮算法,標記過程仍然與「標記-清除」算法同樣,但後續步驟不是直接對可回收對象進行清理,而是讓全部存活的對象都向一端移動,而後直接清理掉端邊界之外的內存
- 分代收集算法,「分代收集」(Generational Collection)算法,把Java堆分爲新生代和老年代,這樣就能夠根據各個年代的特色採用最適當的收集算法。
垃圾回收器
- Serial收集器,串行收集器是最古老,最穩定以及效率高的收集器,可能會產生較長的停頓,只使用一個線程去回收。
- ParNew收集器,ParNew收集器其實就是Serial收集器的多線程版本。
- Parallel收集器,Parallel Scavenge收集器相似ParNew收集器,Parallel收集器更關注系統的吞吐量。
- Parallel Old 收集器,Parallel Old是Parallel Scavenge收集器的老年代版本,使用多線程和「標記-整理」算法
- CMS收集器,CMS(Concurrent Mark Sweep)收集器是一種以獲取最短回收停頓時間爲目標的收集器。
- G1收集器,G1 (Garbage-First)是一款面向服務器的垃圾收集器,主要針對配備多顆處理器及大容量內存的機器. 以極高機率知足GC停頓時間要求的同時,還具有高吞吐量性能特徵
GC算法和垃圾回收器算法圖解以及更詳細內容參考 jvm系列(三):GC算法 垃圾收集器
GC分析 命令調優
主要關注點:
GC日誌分析
摘錄GC日誌一部分(前部分爲年輕代gc回收;後部分爲full gc回收):
2016-07-05T10:43:18.093+0800: 25.395: [GC [PSYoungGen: 274931K->10738K(274944K)] 371093K->147186K(450048K), 0.0668480 secs] [Times: user=0.17 sys=0.08, real=0.07 secs]
2016-07-05T10:43:18.160+0800: 25.462: [Full GC [PSYoungGen: 10738K->0K(274944K)] [ParOldGen: 136447K->140379K(302592K)] 147186K->140379K(577536K) [PSPermGen: 85411K->85376K(171008K)], 0.6763541 secs] [Times: user=1.75 sys=0.02, real=0.68 secs]
經過上面日誌分析得出,PSYoungGen、ParOldGen、PSPermGen屬於Parallel收集器。其中PSYoungGen表示gc回收先後年輕代的內存變化;ParOldGen表示gc回收先後老年代的內存變化;PSPermGen表示gc回收先後永久區的內存變化。young gc 主要是針對年輕代進行內存回收比較頻繁,耗時短;full gc 會對整個堆內存進行回城,耗時長,所以通常儘可能減小full gc的次數
young gc 日誌:
Full GC日誌:
調優命令
Sun JDK監控和故障處理命令有jps jstat jmap jhat jstack jinfo
- jps,JVM Process Status Tool,顯示指定系統內全部的HotSpot虛擬機進程。
- jstat,JVM statistics Monitoring是用於監視虛擬機運行時狀態信息的命令,它能夠顯示出虛擬機進程中的類裝載、內存、垃圾收集、JIT編譯等運行數據。
- jmap,JVM Memory Map命令用於生成heap dump文件
- jhat,JVM Heap Analysis Tool命令是與jmap搭配使用,用來分析jmap生成的dump,jhat內置了一個微型的HTTP/HTML服務器,生成dump的分析結果後,能夠在瀏覽器中查看
- jstack,用於生成java虛擬機當前時刻的線程快照。
- jinfo,JVM Configuration info 這個命令做用是實時查看和調整虛擬機運行參數。
詳細的命令使用參考這裏jvm系列(四):jvm調優-命令篇
調優工具
經常使用調優工具分爲兩類,jdk自帶監控工具:jconsole和jvisualvm,第三方有:MAT(Memory Analyzer Tool)、GChisto。
- jconsole,Java Monitoring and Management Console是從java5開始,在JDK中自帶的java監控和管理控制檯,用於對JVM中內存,線程和類等的監控
- jvisualvm,jdk自帶全能工具,能夠分析內存快照、線程快照;監控內存變化、GC變化等。
- MAT,Memory Analyzer Tool,一個基於Eclipse的內存分析工具,是一個快速、功能豐富的Java heap分析工具,它能夠幫助咱們查找內存泄漏和減小內存消耗
- GChisto,一款專業分析gc日誌的工具
工具使用參考 jvm系列(七):jvm調優-工具篇