JVM類載入過程詳細分析

来源:https://www.cnblogs.com/fourther/archive/2020/04/12/12687964.html
-Advertisement-
Play Games

雙親委派載入模型 為什麼需要雙親委派載入模型 主要是為了安全,避免用戶惡意載入破壞 正常運行的位元組碼文件,比如說載入一個自己寫的 。這樣就有可能造成包衝突問題。 類載入器種類 啟動類載入器:用於載入 中`rt.jar`的位元組碼文件 擴展類載入器:用於載入 中`/jre/lib/ext`文件夾下的位元組 ...


雙親委派載入模型

為什麼需要雙親委派載入模型

主要是為了安全,避免用戶惡意載入破壞JVM正常運行的位元組碼文件,比如說載入一個自己寫的java.util.HashMap.class。這樣就有可能造成包衝突問題。

類載入器種類

file

  • 啟動類載入器:用於載入jdkrt.jar的位元組碼文件
  • 擴展類載入器:用於載入jdk/jre/lib/ext文件夾下的位元組碼文件
  • 應用程式類載入器:載入classPath下的位元組碼文件
  • 自定義類載入器:用戶在程式中自己定義的載入器

源碼分析

1、ClassLoader.loadClass()

    protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
    	// 加鎖
        synchronized (getClassLoadingLock(name)) {
            // First, check if the class has already been loaded
            Class<?> c = findLoadedClass(name);
            // 如果這個Class對象還沒有被載入,下麵就準備載入
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                	// 查看當前類載入器有沒有父類載入器
                    if (parent != null) {
                    	// 父類載入器來載入位元組碼文件
                        c = parent.loadClass(name, false);
                    } else {
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }
				// 如果父類載入器也沒有載入這個Class對象,就由自己來載入
                if (c == null) {
                    // If still not found, then invoke findClass in order
                    // to find the class.
                    long t1 = System.nanoTime();
                    c = findClass(name);

                    // this is the defining class loader; record the stats
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }

不遵守雙親委派載入模型的例子

雙親委派載入模型僅僅是一個約定,後面實現類載入器時,是可以不遵守這個約定。ClassLoader是在JDK1.0的時候就設計好的,而雙親委派載入模型在JDK1.2引入的。所以,有些機制是沒有遵守這個約定的。比如:Service Provider Interface機制的JDBC就沒有遵守這個約定。

1、為什麼JDBC無法遵守這個約定?
JDBCSPI機制的一個例子,JDK定義了java.sql.Connection核心介面,後續MySQLOracle為其提供實現類。在運行中是通過java.sql.DriverManager來獲取指定實現類的實例。這裡需要明白三個問題:

  • java.sql.DriverManager是在rt.jar中,由核心類載入器載入的;
  • 第三方所提供Collection的實現類都是在classpath中;
  • 類中方法想載入新的位元組碼文件時,其初始類載入器就是當前這個類的定義類載入器;

也就是說當JVMjava.sql.DriverManager類的getConnection()方法中獲取Collection實現類的位元組碼時,當前類的定義類載入器是啟動類載入器,而按照約定啟動類載入器是不允許載入classpath下的位元組碼。所以,JDBC就無法遵守這個約定。

2、JDBC是如何解決上面的問題的?
為瞭解決這個,java線上程中放入一個類載入器Thread.currentThread().getContextClassLoader();而這個類載入器可以是隨意的。比如你想載入classpath包下的位元組碼文件,只需要設置當前線程的類載入器為應用程式類載入器即可。

JVM類載入過程

JVM本質的工作就是讀取位元組碼文件、執行位元組碼文件中的指令。其中JVM將讀取位元組碼文件的過程稱為JVM類載入過程。

JVM讀取的位元組碼文件將放在方法區里;

JVM類載入機制分為五個部分:載入、驗證、準備、解析、初始化。如下圖所示:
file

一、Loading:載入

這一步是將JVM外的位元組碼文件載入到JVM內部方法區中的Class對象。

JVM可以通過幾種方式來載入外部的位元組碼文件?

  • 從本地讀位元組碼文件;
  • 從網路讀取位元組碼文件;
  • 通過動態生成的位元組碼文件;

初始類載入器和定義類載入器

由於雙親委派載入模型的存在,一個Class對象的初始類載入器initiating class loader和定義類載入器defining class loader有可能不是同一個。

  • 初始類載入器:它是指讓JVM載入這個位元組碼文件
  • 定義類載入器:它是真正調用defineClass方法,將位元組碼轉換成Class對象

java在判斷instanceof時,只有類名、defining class loader都相等,才表示是同一個類的實例。

Class.getClassLoader()得到的是定義類載入器

相關實驗代碼

1、驗證使用不同ClassLoader載入位元組碼文件

// 這種方法是不遵守雙親委派載入模型的約定
public class ClassLoaderLoading {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        // 這個Class對象是由當前方法的類載入器載入
        Class c1 = MiniJVM.class;
        Class c2 = new MyClassLoader().loadClass("com.github.hcsp.MiniJVM");
        // 使用c2創建一個MiniJVM實例
        Object o = c2.getConstructor().newInstance();
        System.out.println(o instanceof MiniJVM);
        MiniJVM demo = (MiniJVM) o;
    }

    private static class MyClassLoader extends ClassLoader {
        @Override
        public Class<?> loadClass(String name) throws ClassNotFoundException {
            if (name.contains("MiniJVM")) {
                try {
                    byte[] bytes = Files.readAllBytes(new File("target/classes/com/github/hcsp/MiniJVM.class").toPath());
                    return defineClass(name, bytes, 0, bytes.length);
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            } else {
                return super.loadClass(name);
            }
        }
    }
}

2、實現一個遵守雙親委派載入模型的類載入器

public class ClassLoaderLoading {
    public static void main(String[] args) throws ClassNotFoundException {
        Class c1 = MiniJVM.class;
        Class c2 = new MyClassLoader(ClassLoader.getSystemClassLoader()).loadClass("com.github.hcsp.MiniJVM");
        System.out.println("c2 = " + c2);
    }

    private static class MyClassLoader extends ClassLoader {
        public MyClassLoader(ClassLoader systemClassLoader) {
            super(systemClassLoader);
        }
        
        @Override
        public Class<?> loadClass(String name) throws ClassNotFoundException {
            // 載入你想讓這個類載入器載入的位元組碼文件
            if (name.contains("MiniJVM")) {
                try {
                    byte[] bytes = Files.readAllBytes(new File("target/classes/com/github/hcsp/MiniJVM.class").toPath());
                    return defineClass(name, bytes, 0, bytes.length);
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            } else {
                // 其他的位元組碼文件交由父類載入器載入
                return super.loadClass(name);
            }
        }
    }
}

二、Linking:鏈接

當一個.java文件編譯成.class文件時,裡面含有一個符號引用,比如/java/utils/HashMapLinking是指將這符號引用與具體的class對象鏈接起來。

每個位元組碼結構都有一個運行時常量池,它會存儲每個符號引用和所對應的具體對象,以此實現鏈接。

  • Verification:驗證位元組碼的正確性
  • Preparation:為static成員賦預設初始值
  • Resolution:解析當前位元組碼里包含的其他符號引用

三、Initializing

執行初始化方法。比如下麵的四個虛擬機指令:newgetstaticputstaticinvokestatic

原博客地址


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

-Advertisement-
Play Games
更多相關文章
  • 思維 我們只需看與根節點直接相連的邊權權值是1的有幾條,就可判斷以該節點為根節點而開始游戲的勝者,奇數->先手勝 偶數->後手勝。 ...
  • 梯度提升樹(GBDT)的全稱是Gradient Boosting Decision Tree。GBDT還有很多的簡稱,例如GBT(Gradient Boosting Tree), GTB(Gradient Tree Boosting ),GBRT(Gradient Boosting Regressi ...
  • 方法一(手動取cookie) 臨時用一次時,直接將 cookie 複製到 headers 里 方法二(selenium取cookie) 用 selenium 登錄,取 cookie 保存,再添加到 requests 中使用 再次用到時,再陸續更新... 方法三(requests取cookie) ...
  • 被static修飾的成員屬於類,不屬於對象。static修飾的成員被多個對象共用。 定義和使用格式 類變數 static 數據類型 變數名; static int num = 5; 該類的每個對象都"共用"同一個類變數的值。任何對象都可以更改該類變數的值,但也可以在不創建該類的對象的情況下對類變數進 ...
  • 一、synchronized關鍵字 1.我們修改一下上一次連載中的withdraw方法 //synchronized關鍵字添加到成員方法上去可以達到同步記憶體變數的目的 public synchronized void withdraw(double money) { double after = t ...
  • # -*- coding: utf-8 -*-#/usr/local/bin/python3# @Time : 2020/3/7 4:05 PM# @Author : eric# @File : get_audio_loudness.py# @Software: PyCharmimport osim ...
  • 架構概述 B/S架構(Browser/Server,瀏覽器/伺服器模式):是一種通過將瀏覽器作為客戶端的網路結構模式,利用已經逐步成熟的web瀏覽器技術,結合瀏覽器的多種功能,使用瀏覽器來作為早先C/S(Client/Serve)架構下複雜的客戶端,使用C/S架構使得用戶的客戶端得到統一,將軟體系統 ...
  • CILI123磁力導航是資源多,更新快的磁力鏈接搜索導航,集成了10多個有效可用的磁力網站。這些網站包含有幾千萬的影視音樂、軟體、電子書等BT種子資源。更新快的磁力鏈接搜索引擎,實時通過DHT網路獲取最新的BT種子文件信息,並生成磁力鏈接。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...