市面上各類 JVM 相關的資料雖多,但是明顯存在兩個極端:過于生澀難懂,或者流于某個技巧點而不系統化,同時各大公司也都越來越重視推動和發展 JVM 相關技術,一線大廠技術面試現在 JVM 知識也是必考科目,
在這個背景下,我全面梳理了系統化學習 JVM 的知識和經驗,包括 JVM 的技術和記憶體模型,JVM 引數和內置工具,GC 演算法,GC 日志、記憶體和執行緒等相關問題排查分析,以及常見的面試問題深度剖析等高級的進階方法與實戰,既滿足大家快速系統化學習和全面掌握知識的需求,又兼顧大家的面試經驗輔導, 需要的可以點擊這里!!!暗號博客園
JVM面試題
1. 記憶體模型以及磁區,需要詳細到每個區放什么,
2. 堆里面的磁區:Eden,survival (from+ to),老年代,各自的特點,
3. 物件創建方法,物件的記憶體分配,物件的訪問定位,
4. GC 的兩種判定方法:
5. SafePoint 是什么
6. GC 的三種收集方法:標記清除、標記整理、復制演算法的原理與特點,分別用在什么地方,如果讓你優化收集方法,有什么思路?
7. GC 收集器有哪些?CMS 收集器與 G1 收集器的特點,
8. Minor GC 與 Full GC 分別在什么時候發生?
9. 幾種常用的記憶體除錯工具:jmap、jstack、jconsole、jhat
10. 類加載的幾個程序:
11. JVM 記憶體分哪幾個區,每個區的作用是什么?
12. 如和判斷一個物件是否存活?(或者 GC 物件的判定方法)
13. 簡述 java 垃圾回識訓制?
14. java 中垃圾收集的方法有哪些?
15. java 記憶體模型
16. java 類加載程序?
17. 簡述 java 類加載機制?
18. 類加載器雙親委派模型機制?
19. 什么是類加載器,類加載器有哪些?
20. 簡述 java 記憶體分配與回收策率以及 Minor GC 和Major GC
21.怎么獲取 Java 程式使用的記憶體?堆使用的百分比?
22.Java 中堆和堆疊有什么區別?
23.描述一下 JVM 加載 Class 檔案的原理機制?
24. 什么是tomcat類加載機制?
25.Java 堆的結構是什么樣子的?
26. 什么是GC調優?
27. 簡述各個版本記憶體區域的變化?
28. Java 中會存在記憶體泄漏嗎,簡述一下?
29. 垃圾回收的優點和原理,并考慮 2 種回識訓制?基本原理是什么?
30. 什么是分布式垃圾回收(DGC)?它是如何作業的?
31. 講講你理解的性能評價及測驗指標?
32. 常用的性能優化方式有哪些?
32. 說說分布式快取和一致性哈希?
JVM知識點
**JVM**
(1) 基本概念:
JVM是可運行Java代碼的假想計算機,包括一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收,堆和一個存盤方法域,JVM是運行在作業系統之上的,它與硬體沒有直接的互動,

(2) 運行程序:
我們都知道Java源檔案,通過編譯器,能夠生產相應的.Class檔案,也就是位元組碼檔案,而位元組碼檔案又通過Java虛擬機中的解釋器,編譯成特定機器上的機器碼,也就是如下:①Java源檔案—->編譯器—->位元組碼檔案②位元組碼檔案—->JVM—->機器碼每一種平臺的解釋器是不同的,但是實作的虛擬機是相同的,這也就是Java為什么能夠跨平臺的原因了,當一個程式從開始運行,這時虛擬機就開始實體化了,多個程式啟動就會存在多個虛擬機實體,程式退出或者關閉,則虛擬機實體消亡,多個虛擬機實體之間資料不能共享,

**2.執行緒**
這里所說的執行緒指程式執行程序中的一個執行緒物體,JVM 允許一個應用并發執行多個執行緒,Hotspot JVM 中的Java 執行緒與原生作業系統執行緒有直接的映射關系,當執行緒本地存盤、緩沖區分配、同步物件、堆疊、程式計數器等準備好以后,就會創建一個作業系統原生執行緒,Java 執行緒結束,原生執行緒隨之被回收,作業系統負責調度所有執行緒,并把它們分配到任何可用的CPU 上,當原生執行緒初始化完畢,就會呼叫Java 執行緒的run() 方法,當執行緒結束時,會釋放原生執行緒和Java 執行緒的所有資源,
HotspotJVM 后臺運行的系統執行緒主要有下面幾個:
虛擬機執行緒(VM thread):這個執行緒等待JVM 到達安全點操作出現,這些操作必須要在獨立的執行緒里執行,因為當堆修改無法進行時,執行緒都需要JVM 位于安全點,這些操作的型別有:stop-the-world 垃圾回收、執行緒堆疊dump、執行緒暫停、執行緒偏向鎖(biased locking)解除,
周期性任務執行緒:這執行緒負責定時器事件(也就是中斷),用來調度周期性操作的執行,
GC 執行緒:這些執行緒支持JVM 中不同的垃圾回識訓動,
編譯器執行緒:這些執行緒在運行時將位元組碼動態編譯成本地平臺相關的機器碼,
信號分發執行緒:這個執行緒接收發送到JVM 的信號并呼叫適當的JVM 方法處理
3.JVM記憶體區域

JVM記憶體區域主要分為執行緒私有區域【程式計數器、虛擬機堆疊、本地方法區】、執行緒共享區域【JAVA堆、方法區】、直接記憶體,執行緒私有資料區域生命周期與執行緒相同,
依賴用戶執行緒的啟動/結束而創建/銷毀(在HotspotVM內,每個執行緒都與作業系統的本地執行緒直接映射,因此這部分記憶體區域的存/否跟隨本地執行緒的生/死對應),
執行緒共享區域隨虛擬機的啟動/關閉而創建/銷毀,
直接記憶體并不是JVM運行時資料區的一部分,但也會被頻繁的使用:在JDK 1.4引入的NIO提供了基于Channel與Buffer的IO方式,它可以使用Native函式庫直接分配堆外記憶體,然后使用DirectByteBuffer物件作為這塊記憶體的參考進行操作(詳見:Java I/O擴展),這樣就避免了在Java堆和Native堆中來回復制資料,因此在一些場景中可以顯著提高性能,

3.程式計數器(執行緒私有)
一塊較小的記憶體空間,是當前執行緒所執行的位元組碼的行號指示器,每條執行緒都要有一個獨立的程式計數器,這類記憶體也稱為“執行緒私有”的記憶體,正在執行java方法的話,計數器記錄的是虛擬機位元組碼指令的地址(當前指令的地址),如果還是Native方法,則為空,這個記憶體區域是唯一一個在虛擬機中沒有規定任何OutOfMemoryError情況的區域,
4虛擬機堆疊(執行緒私有)
是描述java方法執行的記憶體模型,每個法在執行的同時都會創建一個堆疊幀(Stack Frame)用于存盤區域變數表、運算元堆疊、動態鏈接、方法出口等資訊,每一個方法從呼叫直至執行完成的程序,就對應著一個堆疊幀在虛擬機堆疊中入堆疊到出堆疊的程序,
堆疊幀(Frame)是用來存盤資料和部分程序結果的資料結構,同時也被用來處理動態鏈接(Dynamic Linking)、方法回傳值和例外分派(Dispatch Exception),堆疊幀隨著方法呼叫而創建,隨著方法結束而銷毀——無論方法是正常完成還是例外完成(拋出了在方法內未被捕獲的例外)都算作方法結束,

5.本地方法區(執行緒私有)
本地方法區和Java Stack作用類似,區別是虛擬機堆疊為執行Java方法服務,而本地方法堆疊則為Native方法服務,如果一個VM實作使用C-linkage模型來支持Native呼叫,那么該堆疊將會是一個C堆疊,但HotSpot VM直接就把本地方法堆疊和虛擬機堆疊合二為一,
6.堆(Heap-執行緒共享)-運行時資料區
是被執行緒共享的一塊記憶體區域,創建的物件和陣列都保存在Java堆記憶體中,也是垃圾收集器進行垃圾收集的最重要的記憶體區域,由于現代VM采用分代收集演算法,因此Java堆從GC的角度還可以細分為:新生代(Eden 區、From Survivor 區和To Survivor 區)和老年代,
7.方法區l永久代(執行緒共享)
即我們常說的永久代(Permanent Generation),用于存盤被JVM加載的類資訊、常量、靜態變數、即時編譯器編譯后的代碼等資料. HotSpot VM把GC分代收集擴展至方法區,即使用Java堆的永久代來實作方法區,這樣HotSpot的垃圾收集器就可以像管理Java堆一樣管理這部分記憶體,而不必為方法區開發專門的記憶體管理器(永久帶的記憶體回收的主要目標是針對常量池的回收和型別的卸載,因此收益一般很小),
運行時常量池(Runtime Constant Pool)是方法區的一部分,Class檔案中除了有類的版本、欄位、方法、介面等描述等資訊外,還有一項資訊是常量池
(Constant Pool Table),用于存放編譯期生成的各種字面量和符號參考,這部分內容將在類加載后存放到方法區的運行時常量池中,Java 虛擬機對Class檔案的每一部分(自然也包括常量池)的格式都有嚴格的規定,每一個位元組用于存盤哪種資料都必須符合規范上的要求,這樣才會被虛擬機認可、裝載和執行,
8.JVM運行時記憶體
Java堆從GC的角度還可以細分為:新生代(Eden區、From Survivor區和To Survivor區)和老年代,

2.3.1.新生代
是用來存放新生的物件,一般占據堆的1/3空間,由于頻繁創建物件,所以新生代會頻繁觸發MinorGC進行垃圾回收,新生代又分為Eden區、ServivorFrom、ServivorTo三個區,
1.Eden區Java新物件的出生地(如果新創建的物件占用記憶體很大,則直接分配到老年代),當Eden區記憶體不夠的時候就會觸發MinorGC,對新生代區進行一次垃圾回收,
2.ServivorFrom上一次GC的幸存者,作為這一次GC的被掃描者,
3.ServivorTo保留了一次MinorGC程序中的幸存者,
4.MinorGC的程序(復制->清空->互換)MinorGC采用復制演算法,
1: eden、servicorFrom復制到ServicorTo,年齡+1
首先,把Eden和ServivorFrom區域中存活的物件復制到ServicorTo區域(如果有物件的年齡以及達到了老年的標準,則賦值到老年代區),同時把這些物件的年齡+1(如果ServicorTo不夠位置了就放到老年區);
2:清空eden、servicorFrom
然后,清空Eden和ServicorFrom 中的物件;
3: ServicorTo和 ServicorFrom互換
最后,ServicorTo和ServicorFrom互換,原ServicorTo成為下一次GC時的ServicorFrom區,
9.老年代
主要存放應用程式中生命周期長的記憶體物件,
老年代的物件比較穩定,所以MajorGC 不會頻繁執行,在進行MajorGC前一般都先進行了一次 MinorGC,使得有新生代的物件晉身入老年代,導致空間不夠用時才觸發,當無法找到足夠大的連續空間分配給新創建的較大物件時也會提前觸發一次MajorGC進行垃圾回收騰出空間,
MajorGC采用標記清除演算法:首先掃描一次所有老年代,標記出存活的物件,然后回收沒有標記的物件,MajorGC的耗時比較長,因為要掃描再回收,MajorGC 會產生記憶體碎片,為了減少記憶體損耗,我們一般需要進行合并或者標記出來方便下次直接分配,當老年代也滿了裝不下的時候,就會拋出OOM (Out of Memory)例外,
10.永久代
指記憶體的永久保存區域,主要存放Class 和Meta(元資料)的資訊,Class在被加載的時候被放入永久區域,它和和存放實體的區域不同,GC不會在主程式運行期對永久區域進行清理,所以這也導致了永久代的區域會隨著加載的Class的增多而脹滿,最終拋出OOM例外,
11.垃圾回收與演算法

12.如何確定垃圾
1.參考計數法
在Java 中,參考和物件是有關聯的,如果要操作物件則必須用參考進行,因此,很顯然一個簡單的辦法是通過參考計數來判斷一個物件是否可以回收,簡單說,即一個物件如果沒有任何與之關聯的參考,即他們的參考計數都不為О,則說明物件不太可能再被用到,那么這個物件就是可回收物件,
2.可達性分析
為了解決參考計數法的回圈參考問題,Java 使用了可達性分析的方法,通過一系列的“GC roots"物件作為起點搜索,如果在“GC roots”和一個物件之間沒有可達路徑,則稱該物件是不可達的,要注意的是,不可達物件不等價于可回收物件,不可達物件變為可回收物件至少要經過兩次標記程序,兩次標記后仍然是可回收物件,則將面臨回收,
13.標記清除演算法(Mark-Sweep)
最基礎的垃圾回收演算法,分為兩個階段,標注和清除,標記階段標記出所有需要回收的物件,清除階段回收被標記的物件所占用的空間,如圖
從圖中我們就可以發現,該演算法最大的問題是記憶體碎片化嚴重,后續可能發生大物件不能找到可利用空間的問題,
總結
所有的面試題目都不是一成不變的,特別是像一線大廠,上面的面試真題只是給大家一個借鑒作用,最主要的是給自己增加知識的儲備,有備無患,
給大家分享整理的2019年大廠JVM面試題資料(20多頁pdf檔案)以及多家公司java面試題資料100多頁pdf檔案和各知識點學習路線思維腦圖(xmind)還有JVM講解視頻,需要的可以點擊這里!!!暗號博客園

轉載請註明出處,本文鏈接:https://www.uj5u.com/qita/230204.html
標籤:其他
上一篇:程式員學編程如何入門、進階?一位碼農從四個方面談談他的理解
下一篇:努力就一定會有進步嗎?
