Java程序員必備基礎圖

前言

最近看了深刻理解Java虛擬機第三版,整理了一些基礎結構圖,算是比較全的了,作一下筆記,你們一塊兒學習。java

1.Java虛擬機運行時數據區圖

JVM內存結構是Java程序員必須掌握的基礎。

程序計數器程序員

  • 程序計數器,能夠看做當前線程所執行的字節碼的行號指示器
  • 它是線程私有的。

Java虛擬機棧面試

  • 線程私有的,生命週期與線程相同。
  • 每一個方法被執行的時候都會建立一個"棧幀",用於存儲局部變量表(包括參數)、操做數棧、動態連接、方法出口等信息。
  • 局部變量表存放各類基本數據類型boolean、byte、char、short等

本地方法棧算法

  • 與虛擬機棧基本相似,區別在於虛擬機棧爲虛擬機執行的java方法服務,而本地方法棧則是爲Native方法服務。

Java堆數組

  • Java堆是java虛擬機所管理的內存中最大的一塊內存區域,也是被各個線程共享的內存區域,在JVM啓動時建立。
  • 其大小經過-Xms和-Xmx參數設置,-Xms爲JVM啓動時申請的最小內存,-Xmx爲JVM可申請的最大內存。

方法區緩存

  • 它用於存儲虛擬機加載的類信息、常量、靜態變量、是各個線程共享的內存區域。 -能夠經過-XX:PermSize 和 -XX:MaxPermSize 參數限制方法區的大小。

2. 堆的默認分配圖

  • Java堆 = 老年代 + 新生代
  • 新生代 = Eden + S0 + S1
  • 新生代與老年代默認比例的值爲 1:2 ,能夠經過參數 –XX:NewRatio 配置。
  • 默認的,Eden : from : to = 8 : 1 : 1 ,能夠經過參數–XX:SurvivorRatio 來設定

3.方法區結構圖

方法區是各個線程共享的內存區域,它用於存儲已被虛擬機加載的類型信息、常量、靜態變量、即時編譯器編譯後的代碼緩存等數據。安全

4.對象的內存佈局圖

一個Java對象在堆內存中包括對象頭、實例數據和補齊填充3個部分:數據結構

  • 對象頭包括Mark Word(存儲哈希碼,GC分代年齡等) 和 類型指針(對象指向它的類型元數據的指針),若是是數組對象,還有一個保存數組長度的空間
  • 實例數據是對象真正存儲的有效信息,包括了對象的全部成員變量,其大小由各個成員變量的大小共同決定。
  • 對齊填充不是必然存在的,僅僅起佔位符的做用。

5.對象頭的Mark Word圖

  • Mark Word 用於存儲對象自身的運行時數據,如哈希碼(HashCode)、GC分代年齡、鎖狀態標誌、線程持有的鎖、偏向線程 ID、偏向時間戳等。
  • 在32位的HotSpot虛擬機中,若是對象處於未被鎖定的狀態下,那麼Mark Word的32bit空間裏的25位用於存儲對象哈希碼,4bit用於存儲對象分代年齡,2bit用於存儲鎖標誌位,1bit固定爲0,表示非偏向鎖。

6.對象與Monitor關聯結構圖

對象是如何跟monitor有關聯的呢?佈局

一個Java對象在堆內存中包括對象頭,對象頭有Mark word,Mark word存儲着鎖狀態,鎖指針指向monitor地址。這實際上是Synchronized的底層哦~學習

7.Java Monitor的工做機理圖:

Java 線程同步底層就是監視鎖Monitor~,以下是Java Monitor的工做機理圖:

  • 想要獲取monitor的線程,首先會進入_EntryList隊列。
  • 當某個線程獲取到對象的monitor後,進入_Owner區域,設置爲當前線程,同時計數器_count加1。
  • 若是線程調用了wait()方法,則會進入_WaitSet隊列。它會釋放monitor鎖,即將_owner賦值爲null,_count自減1,進入_WaitSet隊列阻塞等待。
  • 若是其餘線程調用 notify() / notifyAll() ,會喚醒_WaitSet中的某個線程,該線程再次嘗試獲取monitor鎖,成功即進入_Owner區域。
  • 同步方法執行完畢了,線程退出臨界區,會將monitor的owner設爲null,並釋放監視鎖。 。

8.建立一個對象內存分配流程圖

  • 對象通常是在Eden區生成。
  • 若是Eden區填滿,就會觸發Young GC。
  • 觸發Young GC的時候,Eden區實現清除,沒有被引用的對象直接被清除。
  • 依然存活的對象,會被送到Survivor區,Survivor =S0+S1.
  • 每次Young GC時,存活的對象複製到未使用的那塊Survivor 區,當前正在使用的另一塊Survivor 區徹底清除,接着交換兩塊Survivor 區的使用狀態。
  • 若是Young GC要移送的對象大於Survivor區上限,對象直接進入老年代。
  • 一個對象不可能一直呆在新生代,若是它通過屢次GC,依然活着,次數超過-XX:MaxTenuringThreshold的閥值,它直接進入老年代。簡言之就是,對象經歷屢次滾滾長江,紅塵世事,終於成爲長者(進入老年代)

9.可達性分析算法斷定對象存活

可達性分析算法是用來判斷一個對象是否存活的~

算法的核心思想:

  • 經過一系列稱爲「GC Roots」的對象做爲起始點,從這些節點開始根據引用關係向下搜索,搜索走過的路徑稱爲「引用鏈」,當一個對象到 GC Roots 沒有任何的引用鏈相連時(從 GC Roots 到這個對象不可達)時,證實此對象不可能再被使用。

10.標記-清除算法示意圖

  • 標記-清除算法是最基礎的垃圾收集算法。
  • 算法分爲兩個階段,標記和清除。
  • 首先標記出須要回收的對象,標記完成後,統一回收掉被標記的對象。
  • 固然能夠反過來,先標記存活的對象,統一回收未被標記的對象。
  • 標記-清除 兩個缺點是,執行效率不穩定和內存空間的碎片化問題~

11.標記-複製算法示意圖

  • 1969年 Fenichel提出「半區複製」,將內存容量劃分對等兩塊,每次只使用一塊。當這一塊內存用完,將還存活的對象複製到另一塊,而後把已使用過的內存空間一次清理掉~
  • 1989年,Andrew Appel提出「Appel式回收」,把新生代劃分爲較大的Eden和兩塊較小的Survivor空間。每次分配內存只使用Eden和其中一塊Survivor空間。發生垃圾收集時,將Eden和Survivor中仍然存活的對象一次性複製到另一塊Survivor空間上。Eden和Survivor比例是8:1~
  • 「半區複製」缺點是浪費可用空間,而且,若是對象存活率高的話,複製次數就會變多,效率也會下降。

12.標記-整理算法示意圖

  • 1974年,Edward 提出「標記-整理」算法,標記過程跟「標記-清除」算法同樣,接着讓全部存活的對象都向內存空間一端移動,而後直接清理掉邊界之外的內存~
  • 標記-清除算法和標記整理算法本質差別是:前者是一種非移動式的回收算法,後者是移動式的回收算法。
  • 是否移動存活對象都存在優缺點,移動雖然內存回收複雜,可是從程序吞吐量來看,更划算;不移動時內存分配更復雜,可是垃圾收集的停頓時間會更短,因此看收集器取捨問題~
  • Parallel Scavenge收集器是基於標記-整理算法的,由於關注吞吐。CMS收集器是基於標記-清除算法的,由於它關注的是延遲。

13.垃圾收集器組合圖

  • 新生代收集器:Serial、ParNew、Parallel Scavenge
  • 老年代收集器:CMS、Serial Old、Parallel Old
  • 混合收集器:G1

14.類的生命週期圖

一個類從被加載到虛擬機內存開始,到卸載出內存爲止,這個生命週期經歷了七個階段:加載、驗證、準備、解析、初始化、使用、卸載。

加載階段:

  • 經過一個類的全限定名來獲取定義此類的二進制字節流。
  • 將這個字節流所表明的靜態存儲結構轉化爲方法區的運行時數據結構。
  • 在內存中生成一個表明這個類的java.lang.Class對象,做爲方法區這個類的各類數據的訪問入口

驗證:

  • 驗證的目的是確保Class文件的字節流中包含的信息知足約束要求,保證這些代碼運行時不會危害虛擬機自身安全
  • 驗證階段有:文件格式校驗、元數據校驗、字節碼校驗、符號引用校驗。

準備

  • 準備階段是正式爲類中定義的變量(靜態變量)分配內存並設置類變量初始值的階段。

解析

  • 解析階段是虛擬機將常量池內的符號引用替換爲直接引用的過程。

初始化

  • 到了初始化階段,才真正開始執行類中定義的Java字節碼。

15.類加載器雙親委派模型圖

雙親委派模型構成

啓動類加載器,擴展類加載器,應用程序類加載器,自定義類加載器

雙親委派模型工做過程是

若是一個類加載器收到類加載的請求,它首先不會本身去嘗試加載這個類,而是把這個請求委派給父類加載器完成。每一個類加載器都是如此,只有當父加載器在本身的搜索範圍內找不到指定的類時(即ClassNotFoundException),子加載器纔會嘗試本身去加載。

爲何須要雙親委派模型?

若是沒有雙親委派,那麼用戶是否是能夠本身定義一個java.lang.Object的同名類,java.lang.String的同名類,並把它放到ClassPath中,那麼類之間的比較結果及類的惟一性將沒法保證,所以,雙親委派模型能夠防止內存中出現多份一樣的字節碼。

16.棧幀概念結構圖

棧幀是用於支持虛擬機進行方法調用和方法執行背後的數據結構。棧幀存儲了方法的局部變量表、操做數棧、動態鏈接和方法返回地址信息。

局部變量表

  • 是一組變量值的存儲空間,用於存放方法參數和方法內部定義的局部變量。
  • 局部變量表的容量以變量槽(Variable Slot)爲最小單位。

操做數棧

  • 操做數棧,也稱操做棧,是一個後入先出棧。
  • 當一個方法剛剛開始執行的時候, 該方法的操做數棧也是空的, 在方法的執行過程當中, 會有各類字節碼指令往操做數棧中寫入和提取內容, 也就是出棧與入棧操做。

動態鏈接

  • 每一個棧幀都包含一個指向運行時常量池中該棧幀所屬方法的引用, 持有引用是爲了支持方法調用過程當中的動態鏈接(Dynamic Linking)。

方法返回地址

  • 當一個方法開始執行時, 只有兩種方式退出這個方法 。一種是執行引擎遇到任意一個方法返回的字節碼指令。另一種退出方式是在方法執行過程當中遇到了異常。

17.Java內存模型圖

  • Java內存模型規定了全部的變量都存儲在主內存中
  • 每條線程還有本身的工做內存
  • 線程的工做內存中保存了該線程中是用到的變量的主內存副本拷貝
  • 線程對變量的全部操做都必須在工做內存中進行,而不能直接讀寫主內存。
  • 不一樣的線程之間也沒法直接訪問對方工做內存中的變量,線程間變量的傳遞均須要本身的工做內存和主存之間進行數據同步進行。

18.線程狀態轉換關係圖

Java語言定義了6種線程池狀態:

  • 新建(New):建立後還沒有啓動的線程處於這種狀態
  • 運行(Running):線程開啓start()方法,會進入該狀態。
  • 無限等待(Waiting):處於這種狀態的線程不會被分配處理器執行時間,通常LockSupport::park(),沒有設置了Timeoout的Object::wait()方法,會讓線程陷入無限等待狀態。
  • 限期等待(Timed Waiting):處於這種狀態的線程不會被分配處理器執行時間,在必定時間以後他們會由系統自動喚醒。sleep()方法會進入該狀態~
  • 阻塞(Blocked):在程序等待進入同步區域的時候,線程將進入這種狀態~
  • 結束(Terminated):已終止線程的線程狀態,線程已經結束執行

19. Class文件格式圖

  • u一、u二、u四、u8 分別表明1個字節、2個字節、4個字節和8個字節的無符號數
  • 表是由多個無符號數或者其餘表做爲數據項構成的複合數據類型
  • 每一個Class文件的頭四個字節被稱爲魔數(記得之前校招面試,面試官問過我什麼叫魔數。。。)
  • minor和major version表示次版本號,主版本號
  • 緊接着主次版本號以後,是常量池入口,常量池能夠比喻爲Class文件裏的資源倉庫~

20.JVM參數思惟導圖

JVM調優是通往高級開發的必經橋樑,因此好好積累JVM參數配置哈~

我的公衆號

  • 若是你是個愛學習的好孩子,能夠關注我公衆號,一塊兒學習討論。
  • 若是你以爲本文有哪些不正確的地方,能夠評論,也能夠關注我公衆號,私聊我,你們一塊兒學習進步哈。
相關文章
相關標籤/搜索