1. 程式人生 > >java常見面試題

java常見面試題

11.JVM記憶體分哪幾個區,每個區的作用是什麼? java虛擬機器主要分為以下一個區:

方法區:

  1. 有時候也成為永久代,在該區內很少發生垃圾回收,但是並不代表不發生GC,在這裡進行的GC主要是對方法區裡的常量池和對型別的解除安裝
  2. 方法區主要用來儲存已被虛擬機器載入的類的資訊、常量、靜態變數和即時編譯器編譯後的程式碼等資料。
  3. 該區域是被執行緒共享的。
  4. 方法區裡有一個執行時常量池,用於存放靜態編譯產生的字面量和符號引用。該常量池具有動態性,也就是說常量並不一定是編譯時確定,執行時生成的常量也會存在這個常量池中。

虛擬機器棧:

  1. 虛擬機器棧也就是我們平常所稱的棧記憶體,它為java方法服務,每個方法在執行的時候都會建立一個棧幀,用於儲存區域性變量表、運算元棧、動態連結和方法出口等資訊。
  2. 虛擬機器棧是執行緒私有的,它的生命週期與執行緒相同。
  3. 區域性變量表裡儲存的是基本資料型別、returnAddress型別(指向一條位元組碼指令的地址)和物件引用,這個物件引用有可能是指向物件起始地址的一個指標,也有可能是代表物件的控制代碼或者與物件相關聯的位置。區域性變數所需的記憶體空間在編譯器間確定 4.運算元棧的作用主要用來儲存運算結果以及運算的運算元,它不同於區域性變量表通過索引來訪問,而是壓棧和出棧的方式 5.每個棧幀都包含一個指向執行時常量池中該棧幀所屬方法的引用,持有這個引用是為了支援方法呼叫過程中的動態連線.動態連結就是將常量池中的符號引用在執行期轉化為直接引用。

本地方法棧 本地方法棧和虛擬機器棧類似,只不過本地方法棧為Native方法服務。

堆 java堆是所有執行緒所共享的一塊記憶體,在虛擬機器啟動時建立,幾乎所有的物件例項都在這裡建立,因此該區域經常發生垃圾回收操作。

程式計數器 記憶體空間小,位元組碼直譯器工作時通過改變這個計數值可以選取下一條需要執行的位元組碼指令,分支、迴圈、跳轉、異常處理和執行緒恢復等功能都需要依賴這個計數器完成。該記憶體區域是唯一一個java虛擬機器規範沒有規定任何OOM情況的區域。

12.如和判斷一個物件是否存活?(或者GC物件的判定方法) 判斷一個物件是否存活有兩種方法:

  1. 引用計數法 所謂引用計數法就是給每一個物件設定一個引用計數器,每當有一個地方引用這個物件時,就將計數器加一,引用失效時,計數器就減一。當一個物件的引用計數器為零時,說明此物件沒有被引用,也就是“死物件”,將會被垃圾回收. 引用計數法有一個缺陷就是無法解決迴圈引用問題,也就是說當物件A引用物件B,物件B又引用者物件A,那麼此時A,B物件的引用計數器都不為零,也就造成無法完成垃圾回收,所以主流的虛擬機器都沒有采用這種演算法。

2.可達性演算法(引用鏈法) 該演算法的思想是:從一個被稱為GC Roots的物件開始向下搜尋,如果一個物件到GC Roots沒有任何引用鏈相連時,則說明此物件不可用。 在java中可以作為GC Roots的物件有以下幾種:

虛擬機器棧中引用的物件 方法區類靜態屬性引用的物件 方法區常量池引用的物件 本地方法棧JNI引用的物件 雖然這些演算法可以判定一個物件是否能被回收,但是當滿足上述條件時,一個物件比不一定會被回收。當一個物件不可達GC Root時,這個物件並 不會立馬被回收,而是出於一個死緩的階段,若要被真正的回收需要經歷兩次標記 如果物件在可達性分析中沒有與GC Root的引用鏈,那麼此時就會被第一次標記並且進行一次篩選,篩選的條件是是否有必要執行finalize()方法。當物件沒有覆蓋finalize()方法或者已被虛擬機器呼叫過,那麼就認為是沒必要的。 如果該物件有必要執行finalize()方法,那麼這個物件將會放在一個稱為F-Queue的對佇列中,虛擬機器會觸發一個Finalize()執行緒去執行,此執行緒是低優先順序的,並且虛擬機器不會承諾一直等待它執行完,這是因為如果finalize()執行緩慢或者發生了死鎖,那麼就會造成F-Queue佇列一直等待,造成了記憶體回收系統的崩潰。GC對處於F-Queue中的物件進行第二次被標記,這時,該物件將被移除”即將回收”集合,等待回收。

13.簡述java垃圾回收機制? 在java中,程式設計師是不需要顯示的去釋放一個物件的記憶體的,而是由虛擬機器自行執行。在JVM中,有一個垃圾回收執行緒,它是低優先順序的,在正常情況下是不會執行的,只有在虛擬機器空閒或者當前堆記憶體不足時,才會觸發執行,掃面那些沒有被任何引用的物件,並將它們新增到要回收的集合中,進行回收。

14.java中垃圾收集的方法有哪些? 1.標記-清除: 這是垃圾收集演算法中最基礎的,根據名字就可以知道,它的思想就是標記哪些要被回收的物件,然後統一回收。這種方法很簡單,但是會有兩個主要問題:1.效率不高,標記和清除的效率都很低;2.會產生大量不連續的記憶體碎片,導致以後程式在分配較大的物件時,由於沒有充足的連續記憶體而提前觸發一次GC動作。 2.複製演算法: 為了解決效率問題,複製演算法將可用記憶體按容量劃分為相等的兩部分,然後每次只使用其中的一塊,當一塊記憶體用完時,就將還存活的物件複製到第二塊記憶體上,然後一次性清楚完第一塊記憶體,再將第二塊上的物件複製到第一塊。但是這種方式,記憶體的代價太高,每次基本上都要浪費一般的記憶體。 於是將該演算法進行了改進,記憶體區域不再是按照1:1去劃分,而是將記憶體劃分為8:1:1三部分,較大那份記憶體交Eden區,其餘是兩塊較小的記憶體區叫Survior區。每次都會優先使用Eden區,若Eden區滿,就將物件複製到第二塊記憶體區上,然後清除Eden區,如果此時存活的物件太多,以至於Survivor不夠時,會將這些物件通過分配擔保機制複製到老年代中。(java堆又分為新生代和老年代) 3.標記-整理 該演算法主要是為了解決標記-清除,產生大量記憶體碎片的問題;當物件存活率較高時,也解決了複製演算法的效率問題。它的不同之處就是在清除物件的時候現將可回收物件移動到一端,然後清除掉端邊界以外的物件,這樣就不會產生記憶體碎片了。 4.分代收集 現在的虛擬機器垃圾收集大多采用這種方式,它根據物件的生存週期,將堆分為新生代和老年代。在新生代中,由於物件生存期短,每次回收都會有大量物件死去,那麼這時就採用複製演算法。老年代裡的物件存活率較高,沒有額外的空間進行分配擔保,所以可以使用標記-整理 或者 標記-清除。 15.java記憶體模型 java記憶體模型(JMM)是執行緒間通訊的控制機制.JMM定義了主記憶體和執行緒之間抽象關係。執行緒之間的共享變數儲存在主記憶體(main memory)中,每個執行緒都有一個私有的本地記憶體(local memory),本地記憶體中儲存了該執行緒以讀/寫共享變數的副本。本地記憶體是JMM的一個抽象概念,並不真實存在。它涵蓋了快取,寫緩衝區,暫存器以及其他的硬體和編譯器優化。Java記憶體模型的抽象示意圖如下:

從上圖來看,執行緒A與執行緒B之間如要通訊的話,必須要經歷下面2個步驟:

16.java類載入過程? java類載入需要經歷一下7個過程: 載入 載入時類載入的第一個過程,在這個階段,將完成一下三件事情:

  1. 通過一個類的全限定名獲取該類的二進位制流。
  2. 將該二進位制流中的靜態儲存結構轉化為方法去執行時資料結構。
  3. 在記憶體中生成該類的Class物件,作為該類的資料訪問入口。

驗證 驗證的目的是為了確保Class檔案的位元組流中的資訊不回危害到虛擬機器.在該階段主要完成以下四鍾驗證:

  1. 檔案格式驗證:驗證位元組流是否符合Class檔案的規範,如主次版本號是否在當前虛擬機器範圍內,常量池中的常量是否有不被支援的型別.
  2. 元資料驗證:對位元組碼描述的資訊進行語義分析,如這個類是否有父類,是否集成了不被繼承的類等。
  3. 位元組碼驗證:是整個驗證過程中最複雜的一個階段,通過驗證資料流和控制流的分析,確定程式語義是否正確,主要針對方法體的驗證。如:方法中的型別轉換是否正確,跳轉指令是否正確等。
  4. 符號引用驗證:這個動作在後面的解析過程中發生,主要是為了確保解析動作能正確執行。

準備 準備階段是為類的靜態變數分配記憶體並將其初始化為預設值,這些記憶體都將在方法區中進行分配。準備階段不分配類中的例項變數的記憶體,例項變數將會在物件例項化時隨著物件一起分配在Java堆中。

public static int value=123;//在準備階段value初始值為0 。在初始化階段才會變為123 。

1 解析 該階段主要完成符號引用到直接引用的轉換動作。解析動作並不一定在初始化動作完成之前,也有可能在初始化之後。

初始化 初始化時類載入的最後一步,前面的類載入過程,除了在載入階段使用者應用程式可以通過自定義類載入器參與之外,其餘動作完全由虛擬機器主導和控制。到了初始化階段,才真正開始執行類中定義的Java程式程式碼。

  1. 簡述java類載入機制? 虛擬機器把描述類的資料從Class檔案載入到記憶體,並對資料進行校驗,解析和初始化,最終形成可以被虛擬機器直接使用的java型別。

  2. 類載入器雙親委派模型機制? 當一個類收到了類載入請求時,不會自己先去載入這個類,而是將其委派給父類,由父類去載入,如果此時父類不能載入,反饋給子類,由子類去完成類的載入。

19.什麼是類載入器,類載入器有哪些? 實現通過類的許可權定名獲取該類的二進位制位元組流的程式碼塊叫做類載入器。 主要有一下四種類載入器:

  1. 啟動類載入器(Bootstrap ClassLoader)用來載入java核心類庫,無法被java程式直接引用。
  2. 擴充套件類載入器(extensions class loader):它用來載入 Java 的擴充套件庫。Java 虛擬機器的實現會提供一個擴充套件庫目錄。該類載入器在此目錄裡面查詢並載入 Java 類。
  3. 系統類載入器(system class loader):它根據 Java 應用的類路徑(CLASSPATH)來載入 Java 類。一般來說,Java 應用的類都是由它來完成載入的。可以通過 ClassLoader.getSystemClassLoader()來獲取它。
  4. 使用者自定義類載入器,通過繼承 java.lang.ClassLoader類的方式實現。

20.簡述java記憶體分配與回收策率以及Minor GC和Major GC 1.物件優先在堆的Eden區分配。 2.大物件直接進入老年代. 3.長期存活的物件將直接進入老年代. 當Eden區沒有足夠的空間進行分配時,虛擬機器會執行一次Minor GC.Minor Gc通常發生在新生代的Eden區,在這個區的物件生存期短,往往發生Gc的頻率較高,回收速度比較快;Full Gc/Major GC 發生在老年代,一般情況下,觸發老年代GC的時候不會觸發Minor GC,但是通過配置,可以在Full GC之前進行一次Minor GC這樣可以加快老年代的回收速度。