JVM 之類載入

来源:https://www.cnblogs.com/listenfwind/archive/2018/09/25/9695976.html
-Advertisement-
Play Games

一.概述 Java不同於C/C++這類傳統的編譯型語言,也不同於php這一類動態的腳本語言。可以說Java是一種半編譯語言,我們所寫的類會先被編譯成.class文件,這個.class是一串二進位的位元組流。然後當要使用這個類的時候,就會將這個類對應的.class文件載入進記憶體中。而將這個.class的 ...


一.概述

Java不同於C/C++這類傳統的編譯型語言,也不同於php這一類動態的腳本語言。可以說Java是一種半編譯語言,我們所寫的類會先被編譯成.class文件,這個.class是一串二進位的位元組流。然後當要使用這個類的時候,就會將這個類對應的.class文件載入進記憶體中。而將這個.class的內容載入進記憶體,正是通過Jvm類載入機制實現的。

虛擬機把描述類的數據從class文件載入到記憶體,並對數據進行校驗,轉換解析和初始化,最終形成可以被虛擬機直接使用的Java類型,這就是虛擬機的類載入機制。

二.類載入的各個步驟

載入

載入時“類載入”過程的第一步,在載入過程中,虛擬機需要完成以下三件事

  1. 通過一個類的全限定名來獲取定義此類的二進位位元組流。
  2. 將這個位元組流所代表的靜態存儲結構轉化為方法區的運行時數據結構。
  3. 在記憶體中生成一個代表這個類的java.lang.Class對象,作為方法區的這個類的各種數據的訪問入口。

值得一提的是,在載入階段既可以使用系統提供的引導類載入器來完成,也可以由用戶自定義的類載入器來完成,相對而是比較自由的,但對於數組則不是這樣了,數組類本身不通過類載入創建,它是由Java虛擬機直接創建的。但數據所存放的元素類型是需要類載入器去創建的。

載入階段與下一階段的連接部分是交叉進行的,但載入階段和連接階段的開始時間仍然會保持固定的先後順序。

驗證

驗證時連接階段的第一步,這一階段的目的是為了確保Class文件的位元組流中包含的信息複合當前虛擬機的要求,並且不會危害虛擬機自身的安全。雖然說數組越界,將對象胡亂轉型這些操作會被編譯器拒絕編譯,但.class文件並不一定要求從Java源碼編譯而來,可以從其他途徑產生,故而需要對.class文件的二進位流進行驗證。

驗證階段的重要性是不言而喻的,這一階段是否嚴謹,直接決定了Java虛擬機是否能承受惡意代碼的攻擊,從執行性能的角度上講,驗證階段的工作量在虛擬機的類載入系統中又占了相當大的一部分。

從整體上看,驗證階段大致可分為4部分的檢驗動作:文件格式驗證,元數據驗證,位元組碼驗證,符號引用驗證。

  • 符號驗證:主要目的是保證輸入的位元組流能正確地解析並存儲於方法區之內,格式上符合描述一個Java類型信息的要求。這一部分是基於二進位流驗證的,之後會載入到記憶體中,後續驗證是在記憶體中驗證。
  • 元數據驗證:這一驗證主要是對類的元數據信息進行語義校驗,保證不存在不符合Java語言規範的元數據信息。
  • 位元組碼驗證:這一部分是驗證階段中最複雜的一階段,主要目的是通過數據流和控制流分析,確定程式是合法的,符合邏輯的。
  • 符號引用驗證:符號引用是發生在虛擬機將符號引用轉化為直接引用的時候,目的是卻好解析動作能正常執行。

準備

準備階段是為正式類變數(靜態變數)分配記憶體並設置類變數初始值的階段,這些變數所使用的記憶體都講在方法區中進行分配的。值得一提的是,這時候進行分配的僅為類變數(靜態變數),而不包括實例變數。

通常情況下,設置類變數初始值,這個初始值指的是數據類型的預設值,比如int型則是0。但若類變數被final修飾,則情況又不一樣,那樣的話會直接對給定值進行賦值。

解析

解析階段是虛擬機將常量池內的符號引用替換為直接引用的過程。這裡解釋以下什麼是符號引用,什麼是直接引用。

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

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

解析動作主要針對類或介面,欄位,類方法,介面方法,方法類型,方法句柄和調用點限定符7類符號引用進行。

初始化

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

在準備階段,變數已經賦過一次系統要求的初始值,而在初始化階段,則根據程式員通過程式制定的計劃區初始化類變數和其他資源。

三.有意思的代碼段

public class StaticTest
 {

     public static void main(String[] args)
     {
         staticFunction();
     }
  
     static StaticTest st = new StaticTest();
  
     static
     {
         System.out.println("1");
     }
  
     {
         System.out.println("2");
     }
  
     StaticTest()
     {
         System.out.println("3");
         System.out.println("a="+a+",b="+b);
     }
  
     public static void staticFunction(){
         System.out.println("4");
     }
  
     int a=110;
     static int b =112;
 }

這段代碼的運行結果是什麼呢?

答案是:

2

3

a=110,b=0

1

4

這是為什麼呢,大家不妨思考以下。

理解這段代碼不光是要明白Java的類載入機制,還需要明白初始化階段,靜態代碼塊與靜態成員變數的初始化順是與代碼順序有關的。

類載入的過程是:裝載–>連接(驗證,準備,解析)–>初始化。

1.在準備階段,會為類變數設置預設值,所以在案例一中:st=null,b=0,

2.在初始化階段,會先執行類構造器,

換句話說,就是執行static修飾的代碼塊和為static修飾的變數賦值而已。而static修飾的代碼塊和類變數的執行順序是按照它在文件中的先後順序執行的。而static StaticTest st = new StaticTest()排在第一,所以會執行 new StaticTest(),也就是進行對象的初始化

2.1.在對象的初始化過程中,會先執行成員變數(代碼塊),然後再執行構造方法.成員變數的執行順序也是誰先聲明,誰先執行,所以排在第一的代碼塊

2.2成員變數執行完後,執行構造方法.此時,a=110,b=0;

3.由static StaticTest st = new StaticTest();觸發的非靜態代碼的初始化過程到此結束,接下來繼續執行靜態代碼的初始化,於是輸出 1 。

4.整個類載入到此結束,執行代碼,輸出 4 。

再看下一道

public class StaticTest
 {

     public static void main(String[] args)
     {
         staticFunction();
     }
  
  
     static
     {
         System.out.println("1");
     }
  
     {
         System.out.println("2");
     }
  
     StaticTest()
     {
         System.out.println("3");
         System.out.println("a="+a+",b="+b);
     }
  
     public static void staticFunction(){
         System.out.println("4");
     }
  
     int a=110;
     static int b =112;
     static StaticTest st = new StaticTest();  //將這條語句放到最下麵
 }

僅僅是改變一條語句,而這段代碼的運行結果是

1

2

3

a=110,b=112

4

大家不妨運用上面的知識,想想是為什麼。





如果覺得對你有幫助,不如花0.5元請作者吃顆糖,讓他甜一下吧~~


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

-Advertisement-
Play Games
更多相關文章
  • 時通訊:支持好友,群組,發圖片、文件,消息聲音提醒,離線消息,保留聊天記錄 工作流模塊 1.模型管理 :web線上流程設計器、預覽流程xml、導出xml、部署流程 2.流程管理 :導入導出流程資源文件、查看流程圖、根據流程實例反射出流程模型、激活掛起 3.運行中流程:查看流程信息、當前任務節點、當前 ...
  • 1.socket 伺服器搭建 實例化socket伺服器,迴圈獲取請求 SocketThread類實現多線程通信 單例Chatmanage,對所有客戶端線程管控處理 至此,伺服器搭建完成 2.客戶端(創建兩個客戶端) 客戶端1(監聽指定伺服器,通過控制台輸入消息進行伺服器與客戶端以及客戶端之間的通信, ...
  • 如今,支付的引入是很多互聯網產品都需要的。為了讓用戶用著更“舒心”,集成像支付寶、微信支付這樣的第三方支付也就成了常有的事。今天就來看看微信支付,涉及代碼之處均用 Python 編寫。 要想開發順利進行,首先要對業務流程有個清晰的認識。這裡以微信公眾號支付為例,因此也借用微信支付官方文檔中的業務流程 ...
  • java當中throws子句在繼承當中overrride時有什麼規則? ...
  • 以下兩個例子說明synchronized塊的用法: (視頻下載) (全部書籍)例1.9.4_a-本章源碼class A { public void disp() { System.out.println("新線程馬克-to-win啟動:"); for (int i = 0; i < 10; i++) ...
  • 標準庫 map set 刪除 刪除操作 有map如下: 刪除方法: | 刪除操作種類 | 功能描述 | | | | | cnt.erase(3); | 刪除key為3的元素,並返回刪除的元素的個數 | | cnt.erase(p); | p為迭代器,刪除p指向的元素,並返回p之後元素的迭代器 | | ...
  • Java的基礎性數據類型並不算多,基本類型的包裝類以及String BigInteger BigDecimal等,這是平時經常用到的,雖然天天使用,就是因為太基礎所以很少有人系統認真的對這些數據類型進行分析,本文著重從整體的邏輯思路對這些基礎性的類型進行了介紹. ...
  • 下載:https://pan.baidu.com/s/1IakOOvmfltodm6w_taDcQg ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...