深入瞭解Java虛擬機(3-1)虛擬機類載入機制

来源:http://www.cnblogs.com/zhangxinly/archive/2017/08/16/7376391.html
-Advertisement-
Play Games

虛擬機類載入機制 一、類載入的階段和時機 1.階段 整個生命周期包括:載入(Loading)、驗證(Verification)、準備(Preparation)、解析(Resolution)、初始化(Initialization)、使用(Using)和卸載(Unloading)7個階段。 其中驗證、準 ...


 

虛擬機類載入機制

一、類載入的階段和時機

  1.階段

      整個生命周期包括:載入(Loading)、驗證(Verification)、準備(Preparation)、解析(Resolution)、初始化(Initialization)、使用(Using)和卸載(Unloading)7個階段。

      其中驗證、準備、解析3個部分統稱為連接(Linking),

  2.類載入時機

      1)遇到new、getstatic、putstatic或invokestatic這4條位元組碼指令時,如果類沒有進行過初始化,則需要先觸發其初始化。

        常用:使用new關鍵字實例化對象的時候、讀取或設置一個類的靜態欄位(被final修飾、已在編譯期把結果放入常量池的靜態欄位除外)的時候,以及調用一個類的靜態方法的時候。

      2)使用java.lang.reflect包的方法對類進行反射調用的時候,如果類沒有進行過初始化,則需要先觸發其初始化。

      3)當初始化一個類的時候,如果發現其父類還沒有進行過初始化,則需要先觸發其父類的初始化(介面除外)

      4)當虛擬機啟動時,用戶需要指定一個要執行的主類(包含main()方法的那個類),虛擬機會先初始化這個主類。

      5)當使用JDK 1.7的動態語言支持時,如果一個java.lang.invoke.MethodHandle實例最後的解析結果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,並且這個方法句柄所對應的類沒有進行過初始化,則需要先觸發其初始化

    例子

      1.對於靜態欄位,只有直接定義這個欄位的類才會被初始化,因此通過其子類來引用父類中定義的靜態欄位,只會觸發父類的初始化而不會觸發子類的初始化

package org.fenixsoft.classloading;
/**
*被動使用類欄位演示一:
*通過子類引用父類的靜態欄位,不會導致子類初始化
**/
public class SuperClass{
    static{
        System.out.println("SuperClass init!");
    }
    public static int value=123;
}

public class SubClass extends SuperClass{
    static{
        System.out.println("SubClass init!");
    }
}
/*
**
*非主動使用類欄位演示
**/
public class NotInitialization{
    public static void main(String[]args){
        System.out.println(SubClass.value);
    }
}

 

      2.類的數組類型

package org.fenixsoft.classloading;
/**
*被動使用類欄位演示二:
*通過數組定義來引用類,不會觸發此類的初始化
*虛擬機會初始化一個[SuperClass的數組類,由虛擬機自動產生,通過執行newarray位元組碼
**/
public class NotInitialization{
    public static void main(String[]args){
        SuperClass[]sca=new SuperClass[10];
    }
}    

 

      3.常量欄位(static final)

package org.fenixsoft.classloading;
/**
*被動使用類欄位演示三:
*常量在編譯階段會存入調用類的常量池中,本質上並沒有直接引用到定義常量的類,因此不會觸發定義常量的類的初始化。
**/
public class ConstClass{
    static{
        System.out.println("ConstClass init!");
    }
    public static final String HELLOWORLD="hello world";
}
/*
**
*非主動使用類欄位演示
**/
public class NotInitialization{
    public static void main(String[]args){
        System.out.println(ConstClass.HELLOWORLD);
    }
}

 

二、類載入過程

  1.載入

      1)通過一個類的全限定名來獲取定義此類的二進位位元組流。

      2)將這個位元組流所代表的靜態存儲結構轉化為方法區的運行時數據結構。

      3)在記憶體中生成一個代表這個類的java.lang.Class對象,作為方法區這個類的各種數據的訪問入口。

    數組的載入:

      數組的元素類型是基本類型:引導類載入器

      數組的元素類型是引用類型:遞歸父類載入

  2.驗證:-Xverify:none取消

      1)文件格式驗證:驗證class文件格式,並能被當前虛擬機處理

      2)元數據驗證:位元組碼描敘信息進行語義分析,如:類是否有父類,是否與父類欄位衝突

      3)位元組碼驗證:確定程式的語義合乎邏輯

      4)符號引用驗證:符號引用所引用的字元串的驗證,如:全限定名能否找到對應的類,類、欄位、方法的訪問性

  3.準備

    為類變數分配記憶體並設置類變數初始值的階段,這些變數所使用的記憶體都將在方法區中進行分配

    註意:類變數(static)、初始值為類型的初值,而並不是“=”號賦予的值;如:int初值為0等;但是static fianl常量除外(欄位表中ConstantValue屬性指定的值)

    

 

  4.解析

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

      1)符號引用(Symbolic References):符號引用以一組符號來描述所引用的目標,符號可以是任何形式的字面量,只要使用時能無歧義地定位到目標即可。

        符號引用與虛擬機實現的記憶體佈局無關,引用的目標並不一定已經載入到記憶體中。

        各種虛擬機實現的記憶體佈局可以各不相同,但是它們能接受的符號引用必須都是一致的,因為符號引用的字面量形式明確定義在Java虛擬機規範的Class文件格式中。

      2)直接引用(Direct References):直接引用可以是直接指向目標的指針、相對偏移量或是一個能間接定位到目標的句柄。

        直接引用是和虛擬機實現的記憶體佈局相關的,同一個符號引用在不同虛擬機實例上翻譯出來的直接引用一般不會相同。

        如果有了直接引用,那引用的目標必定已經在記憶體中存在。

      解析的類型:類或介面、欄位、類方法、介面方法、方法類型、方法句柄和調用點限定符

      1.類或介面的解析

        假設當前代碼所處的類為D,如果要把一個從未解析過的符號引用N解析為一個類或介面C的直接引用,那虛擬機完成整個解析的過程需要以下3個步驟:

        1)非數組類型,那虛擬機將會把代表N的全限定名傳遞給D的類載入器去載入這個類C。

        2)數組類型,並且數組的元素類型為對象,也就是N的描述符會是類似“[Ljava/lang/Integer”的形式,那將會按照第1點的規則載入數組元素類型。如果N的描述符如前面所假設的形式,需要載入的元素類型就是“java.lang.Integer”,接著由虛擬機生成一個代表此數組維度和元素的數組對象。

        3)如果上面的步驟沒有出現任何異常,那麼C在虛擬機中實際上已經成為一個有效的類或介面了,但在解析完成之前還要進行符號引用驗證,確認D是否具備對C的訪問許可權。如果發現不具備訪問許可權,將拋出java.lang.IllegalAccessError異常。

      2.欄位的解析

        本類----介面----父類遞歸----不存在拋出異常

      3.類方法解析

        解析出的符號引用不是類,拋出異常----本類----父類----介面(抽象方法拋異常)----不存在拋異常

      4.介面方法解析

        解析出非符號引用不是介面,拋異常----本介面----父介面----不存在拋出異常

  4.初始化

      初始化是執行類構造器<clinit>方法

      <clinit>()方法是由編譯器自動收集類中的所有類變數的賦值動作和靜態語句塊(static{}塊)中的語句合併產生的

      編譯器收集的順序是由語句在源文件中出現的順序所決定的,靜態語句塊中只能訪問到定義在靜態語句塊之前的變數,定義在它之後的變數,在前面的靜態語句塊可以賦值,但是不能訪問

 

三、類載入器

  比較兩個類是否“相等”:同一個類載入器,載入同一個類;才是相等

  否則同一個Class文件,被同一個虛擬機載入,只要載入它們的類載入器不同,那這兩個類就必定不相等。

  這裡所指的“相等”,包括代表類的Class對象的equals()方法、isAssignableFrom()方法、isInstance()方法的返回結果,也包括使用instanceof關鍵字做對象所屬關係判定等情況。

  1.類載入器

    啟動類載入器:載入JAVA_HOME\lib下或-Xbootclasspath指定路徑下的指定類庫

    擴展類載入器:載入ext類庫,任何類庫都行

    應用程式類載入器:用戶用的最多的載入器

  2.雙親委派模型

    雙親委派模型:保證了類的相等或者說是唯一

      如果一個類載入器收到了類載入的請求,它首先不會自己去嘗試載入這個類,而是把這個請求委派給父類載入器去完成,每一個層次的類載入器都是如此,

      因此所有的載入請求最終都應該傳送到頂層的啟動類載入器中,只有當父載入器反饋自己無法完成這個載入請求(它的搜索範圍中沒有找到所需的類)時,子載入器才會嘗試自己去載入。

      

 

  3.打破雙親委派模型

    JNDI服務

      JNDI提供命名服務和目錄服務,是通過啟動類載入器載入,是對資源的管理

      但是既然是管理資源,就要載入應用程式中三方廠商提供的JNDI介面,才能對三方資源進行目錄和命名服務提供

      而啟動類載入器是無法載入到這些應用程式中的資源

      解決:通過一個ThreadContextClassLoader線程上下文載入器載入

     OSGi

      熱部署,動態的增加和卸載模塊,原有的類載入機制就不合適了

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • asp.net core微軟官方為日誌提供了原生支持,有如下實現 Console Debug EventLog AzureAppServices TraceSource EventSource 並且在asp.net core項目模板中預設開啟Comsole,Debug的日誌輸出 在實際生產中僅僅是控 ...
  • 摘要:本來說這個企業的門戶網站單純的做做顯示公司文化信息的,做好了老闆說要新增線上辦理業務,本來這個網站是基於別人的框架做的前臺都只能用純html來做。好吧上兩篇我就寫了table裡面向資料庫插入數據《ASP.NET前臺html頁面AJAX提交數據後臺ashx頁面接收數據》跟前臺的查詢數據《ASP.... ...
  • 領域對象:Game(游戲), Room(游戲群),兩者一對多的關係,SQL語句中會用到JOIN 通常用Dapper的Query<TFirst,TSecond,TReturn>() 或 QueryAsync<TFirst,TSecond,TReturn>() 是可以實現,但是去除重覆記錄比較麻煩。 所 ...
  • 一般是訪問https時才出現“508 Loop Detected”,idhttp+IdSSLIOHandlerSocketOpenSSL,這個在上篇文章中講過了。 由於該問題網上資料極少,連外文資料也沒卵用,起初我也以為是idhttp的重定向設置問題,但確認過沒設置錯。 不過相比https,http ...
  • 出現這種問題的原因是由於訪問的 URL地址為https或存在其跳轉地址為https。 首先單純使用idhttp是只能訪問http,而https則需要搭配IdSSLIOHandlerSocketOpenSSL來實現對https的訪問支持,當然還需要在系統目錄或編譯目標程式的同目錄下有ssleay32. ...
  • 安裝好環境後,開始了第一個Hello word 例子,如何讀取圖片,保存圖品 每天學習一點點,睡覺去了。 ...
  • 以前學過點 面向對象的知識,我感覺這之間是有關聯的,比如說裝飾器的第一個要素是對被裝飾的函數的封閉性,不允許更改;第二個就是對裝飾器本oj身的可擴展性。 裝飾器要點:高階函數+嵌套函數=裝飾器 需要掌握的知識點:1、函數即變數 2、高階函數(函數的參數也是函數) 3、嵌套函數 裝飾器一:裝飾器不帶參 ...
  • 今天寫這個是為了 提醒自己 編程過程 不僅要有邏輯 思想 還有要規範 代碼 這樣可讀性 感謝我牛神提供的文檔 1、PHP 編程規範與編碼習慣最主要的有以下幾點: 1 文件說明 2 function 函數體說明 3 代碼縮進 4 if省略 5 變數規範 6 命名規範 7 十行一註釋 8 註釋風格 9 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...