95%的技術面試必考的JVM知識點都在這

jvm的知識點彙總共6個大方向:內存模型、類加載機制、GC垃圾回收是比較重點的內容。性能調優部分偏重實際應用,重點突出實踐能力。編譯器優化和執行模式部分偏重理論基礎,主要掌握知識點。前端

各個部分的內容以下:java

1>內存模型部分:程序計數器、方法區、堆、棧、本地方法棧的做用,保存哪些數據;面試

2>類加載部分:雙親委派的加載機制以及經常使用類加載器分別加載哪一種類型的類;算法

3>GC部分:分代回收的思想和依據,以及不一樣垃圾回收算法實現的思路、適合的場景;編程

4>性能調優部分:經常使用的jvm優化參數的做用,參數調優的依據,要了解經常使用的jvm分析工具能分析哪類問題,以及使用方法;bootstrap

5>執行模式部分:解釋、編譯、混合模式的優缺點,瞭解java7提供的分層編譯技術。須要知道JIT即時編譯技術和OSR也就是棧上替換,知道C一、C2編譯器針對的場景,其中C2針對server模式,優化更激進。在新技術方面能夠了解一下java10提供的由java實現的graal編譯器。安全

6>編譯優化部分:前端編譯器javac的編譯過程、AST抽象語法樹、編譯期優化和運行期優化。編譯優化的經常使用技術,包括公共子表達式的消除、方法內聯、逃逸分析、棧上分配、同步消除等。明白了這些才能寫出對編譯器友好的代碼。多線程

jvm的內容相對來講比較集中,可是對知識深度的掌握要求較高,建議面試前重點增強。併發

1、jvm內存相關考點app

1.詳解-jvm內存模型

jvm內存模型主要指運行時的數據區,包括5個部分。

棧也叫方法棧,是線程私有的,線程在執行每一個方法時都會同時建立一個棧幀,用來存儲局部變量表、操做棧、動態連接、方法出口等信息。調用方法時執行入棧,方法返回時執行出棧。

本地方法棧與棧相似,也是用來保存線程執行方法時的信息,不一樣的是,執行java方法使用棧,而執行native方法使用本地方法棧。

程序計數器保存着當前線程所執行的字節碼位置,每一個線程工做時都有一個獨立的計數器。程序計數器爲執行java方法服務,執行native方法時,程序計數器爲空。

棧、本地方法棧、程序計數器這三個部分都是線程獨佔的。

堆是jvm管理的內存中最大的一塊,堆被全部線程共享,目的是爲了存放對象實例,幾乎全部的對象實例都在這裏分配。當堆內存沒有可用的空間時,會拋出OOM異常。根據對象存活的週期不一樣,jvm把堆內存進行分代管理,由垃圾回收器來進行對象的回收管理。

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

jdk1.7中的永久代和1.8中的metaspace都是方法區的一種實現。

面試回答此知識點相關問題時,要答出兩個要點:一個是各部分的功能,另外一個是哪些線程共享,哪些獨佔。

2.詳解-jmm內存可見性

jmm是java內存模型,與剛纔講到的jvm內存模型是兩回事,jmm的主要目標是定義程序中變量的訪問規則,如圖所示,全部的共享變量都存儲在主內存中共享。每一個線程有本身的工做內存,工做內存中保存的是主內存中變量的副本,線程對變量的讀寫等操做必須在本身 的工做內存中進行,而不能直接讀寫主內存中的變量。

在多線程進行數據交互時,例如線程a給一個共享變量賦值後,由線程b來讀取這個值,a修改完變量是修改在本身的工做區內存中,b是不可見的,只有從a的工做區寫回到主內存,b再從主內存讀取到本身的工做區才能進行進一步的操做。因爲指令重排序的存在,這個寫-讀的順序有可能被打亂。

所以jmm須要提供原子性、可見性、有序性的保證。

三、詳解-jmm保證

主要介紹下jmm如何保證原子性、可見性,有序性。

jmm保證對除long和double外的基礎數據類型的讀寫操做是原子性的。另外關鍵字Synchronized也能夠提供原子性保證。Synchronized的原子性是經過java的兩個高級的字節碼指令monitorenter和monitorexit來保證的。

jmm可見性的保證,一個是經過Synchronized,另一個就是volatile。volatile強制變量的賦值會同步刷新回主內存,強制變量的讀取會從主內存從新加載,保證不一樣的線程老是可以看到該變量的最新值。

jmm對有序性的保證,主要經過volatile和一系列happens-before原則。volatile的另外一個做用就是阻止指令重排序,這樣就能夠保證變量讀寫的有序性。

happens-before原則包括一系列規則,如

程序順序原則,即一個線程內必須保證語義串行性;

鎖規則,即對同一個鎖的解鎖必定發生在再次加鎖以前;

此外還包括happens-before原則的傳遞性、線程啓動、中斷、終止規則等。

2、類加載機制相關考點

1.詳解類加載機制

類的加載指的是將編譯好的class類文件中的字節碼讀入到內存中,將其放在方法區內並建立對應的Class對象。

類的加載分爲加載、連接、初始化,其中連接又包括驗證、準備、解析三步。看到圖中上半部分深綠色,咱們逐個分析:

加載是文件到內存的過程。經過類的徹底限定名查找此類字節碼文件,並利用字節碼文件建立一個Class對象

驗證是對類文件內容驗證。目的在於確保Class文件符合當前虛擬機要求,不會危害虛擬機自身安全。主要包括四種:文件格式驗證,元數據驗證,字節碼驗證,符號引用驗證。

準備階段是進行內存分配。爲類變量也就是類中由static修飾的變量分配內存,而且設置初始值,這裏要注意,初始值是0或者null,而不是代碼中設置的具體值,代碼中設置的值是在初始化階段完成的。另外這裏也不包含用final修飾的靜態變量,由於final在編譯的時候就會分配了。

解析主要是解析字段、接口、方法。主要是將常量池中的符號引用替換爲直接引用的過程。直接引用就是直接指向目標的指針、相對偏移量等。

最後是初始化:主要完成靜態塊執行與靜態變量的賦值。這是類加載最後階段,若被加載類的父類沒有初始化,則先對父類進行初始化。

只有對類主動使用時,纔會進行初始化,初始化的觸發條件包括建立類的實例的時候、訪問類的靜態方法或者靜態變量的時候、Class.forName()反射類的時候、或者某個子類被初始化的時候。

類的生命週期,就是從類的加載到類實例的建立與使用,再到類對象再也不被使用時能夠被GC卸載回收。這裏要注意一點,由java虛擬機自帶的三種類加載器加載的類在虛擬機的整個生命週期中是不會被卸載的,只有用戶自定義的類加載器所加載的類才能夠被卸載。

2.詳解類加載器

java自帶的三種類加載器分別是:bootstrap啓動類加載器、擴展類加載器和應用加載器也叫系統加載器。圖右邊的桔黃色文字表示各種加載器對應的加載目錄。啓動類加載器加載java home中lib目錄下的類,擴展加載器負責加載ext目錄下的類,應用加載器加載classpath指定目錄下的類。

除此以外,能夠自定義類加載器。

java的類加載使用雙親委派模式,即一個類加載器在加載類時,先把這個請求委託給本身的父類加載器去執行,若是父類加載器還存在父類加載器,就繼續向上委託,直到頂層的啓動類加載器,如圖中藍色向上的箭頭。若是父類加載器可以完成類加載,就成功返回,若是父類加載器沒法完成加載,那麼子加載器纔會嘗試本身去加載。

這種雙親委派模式的好處,一個能夠避免類的重複加載,另外也避免了java的核心API被篡改。

3、其餘知識梳理

1.詳解分代回收

前面提到過,java的堆內存被分代管理,分代管理主要是爲了方便垃圾回收,這樣作基於2個事實,第1、大部分對象很快就再也不使用,第二,還有一部分不會當即無用,但也不會持續很長時間。

虛擬機中劃分爲年輕代、老年代、和永久代。

1>年輕代:主要用來存放新建立的對象,年輕代分爲eden區和兩個Survivor區。大部分對象在Eden區中生成。當Eden區滿時,還存活的對象會在兩個Survivor區交替保存,達到必定次數的對象會晉升到老年代。

2>老年代:用來存放從年輕代晉升而來的,存活時間較長的對象。

3>永久代:主要保存類信息等內容,這裏的永久代是指對象劃分方式,不是專指1.7的permGen,或者1.8以後的metaspace。

根據年輕代與老年代的特色,jvm提供了不一樣的垃圾回收算法。垃圾回收算法按類型能夠分爲引用計數法、複製法和標記清除法。

其中引用計數法是經過對象被引用的次數來肯定對象是否被使用,缺點是沒法解決循環引用的問題。

複製算法須要from和to兩塊相同大小的內存空間,對象分配時只在from塊中進行,回收時把存活對象複製到to塊中,並清空from塊,而後交換兩塊的分工,即把from塊做爲to塊,把to塊做爲from塊。缺點是內存使用率較低。

標記清除算法分爲標記對象和清除不在使用的對象兩個階段,標記清除算法的缺點是會產生內存碎片。

jvm中提供的年輕代回收算法Serial、ParNew、Parallel Scavenge都是複製算法,而CMS、G一、zgc都屬於標記清除算法。

本篇文章,對這幾個算法就不展開了。

總結:面試考察點及加分項

1.jvm相關的面試考察點

首先,須要jvm的內存模型和java的內存模型;

其次,要了解的類的加載過程,瞭解雙親委派機制;

第三,要理解內存的可見性與java內存模型對原子性、可見性、有序性的保證機制;

第四,要了解經常使用的gc算法的特色、執行過程,和適用場景,例如g1適合對最大延遲有要求的場合,zgc適用於64爲系統的大內存服務中;

第五,要了解經常使用的jvm參數,明白對不一樣參數的調整會有怎樣的影響,適用什麼樣的場景。例如垃圾回收的併發數、偏向鎖設置等

2.相關加分項

若是想要面試官對你留下更好的印象的話,注意這些加分項:

首先,若是在編譯器優化方面有深刻的瞭解的話,會讓面試官以爲你對技術的深度比較有追求。例如知道在編程時如何合理利用棧上分配下降gc壓力、如何編寫適合內聯優化等代碼等。

其次,若是你能有線上實際問題的排查經驗或思路那就更好了,面試官都喜歡動手能力強的同窗。例如解決過線上常常full gc問題,排查過內存泄露問題等。

第三,若是能有針對特定場景的jvm優化實踐或者優化思路,也會有意想不到的效果。例如針對高併發低延遲的場景,如何調整gc參數儘可能下降gc停頓時間,針對隊列處理機如何儘量提升吞吐率等;

第四,若是對最新的jvm技術趨勢有所瞭解,也會給面試官留下比較深入的印象。例如瞭解zgc高效的實現原理,瞭解Graalvm的特色等。

總之,掌握以上具體的JVM考點,才能在面試時應答自如。但願讀完此篇文章的你,都能在金三銀四的招聘季作好準備,拿到心儀的Offer。

相關文章
相關標籤/搜索