JAVA學習之ClassLoader

来源:https://www.cnblogs.com/waaaafool/archive/2020/03/27/12578914.html
-Advertisement-
Play Games

JAVA學習之ClassLoader 前言 最近被 一句話所觸動—— 種一棵樹最好的時間是十年前,其次是現在。 所以決定要開始記錄自己的學習之路。 什麼是類載入? 我們都知道,每個.java文件可以經過javac指令編譯成.class文件,裡面包含著java虛擬機的機器指令。當我們需要使用一個jav ...


JAVA學習之ClassLoader

前言

最近被 一句話所觸動——種一棵樹最好的時間是十年前,其次是現在。所以決定要開始記錄自己的學習之路。

什麼是類載入?

我們都知道,每個.java文件可以經過javac指令編譯成.class文件,裡面包含著java虛擬機的機器指令。當我們需要使用一個java類時,虛擬機會載入它的.class文件,創建對應的java對象。將.class調入虛擬機的過程,稱之為載入。

loading :載入。通過類的完全限定名找到.class位元組碼文件,同時創建一個對象。

verification:驗證。確保class位元組碼文件符合當前虛擬機的要求。

preparation:準備。這時候將static修飾的變數進行記憶體分配,同時設置初始值。

resolution:解析。虛擬機將常量池中的符號引用變為直接引用。

  • 符號引用:在編譯之後完成的,一個常量並沒有進行記憶體分配,也就只能用符號引用。
  • 直接引用:常量會在preparation階段將常量進行記憶體分配,於是就可以建立直接的虛擬機記憶體聯繫,就可以直接引用。

initialization:初始化。類載入的最後階段。如果這個類有超類,進行超類的初始化,執行類的靜態代碼塊,同時給類的靜態變數賦予初值。前面的preparation階段是分配記憶體,都只是預設的值,並沒有被賦予初值。

類載入在java中有兩種方式

  • 顯示載入。通過class.fornNme(classname)或者this.getClass().getClassLoader.loadClass(classname)載入。即通過調用類載入器classLoader來完成類的載入
  • 隱式載入。類在需要被使用時,不直接調用ClassLoader,而是虛擬機自動調用ClassLoader載入到記憶體中。

什麼是ClassLoader?

類載入器的任務是將類的二進位位元組流讀入到JVM中,然後變成一個JVM能識別的class對象,同時實例化。於是我們就可以分解ClassLoader的任務。

  1. 找到類的二進位位元組流,並載入進來
  2. 規則化為JVM能識別的class對象

我們查看源碼,找到對應解決方案:

在ClassLoader中,定義了兩個介面:

  1. findClass(String name).
  2. defineClass(byte[] b , int off , int len)

findClass用於找到二進位文件並讀入,調用defineClass用位元組流變成JVM能識別的class對象,同時實例化class對象。

我們來看個例子:

protected Class<?> findClass(String name) throws ClassNotFoundException {
	  // 獲取類的位元組數組,通過name找到類,如果你對位元組碼加密,需要自己解密出來
      byte[] classData = getClassData(name);  
      if (classData == null) {
          throw new ClassNotFoundException();
      } else {
	      //使用defineClass生成class對象
          return defineClass(name, classData, 0, classData.length);
      }
  }

代理模式

提到類載入器,一定得涉及的是委派模式。在JAVA中,ClassLoader存在一下幾類,他們的關係如下:

  • Bootstrap ClassLoader:引導類載入器。採用原生c++實現,用於載入java的核心類(%JAVA_HOME%/lib路徑下的核心類庫或者 -Xbootclasspath指定下的jar包)到記憶體中。沒有父類

  • Extension ClassLoader:擴展類載入器。java實現,載入/lib/ext目錄下或者由系統變數-Djava.ext.dir指定位路徑中的類庫。父類載入器為null。

  • System ClassLoader:它會根據java應用的類路徑(CLASSPATH)來載入類,即java -classpath-D java.class.path 指定路徑下的類庫,也就是我們經常用到的classpath路徑。一般來說,java應用的類都是由它完成。可以由ClassLoader.getSystemLoader()方法獲得。父類載入器是Extension ClassLoader。

  • Customize ClassLoader:用戶自定義載入器。用於完成用戶自身的特有需求。父類載入器為System ClassLoader。

    1. 載入不在CLASSPATH路徑的class文件
    2. 加密後的class文件需要用戶自定義ClassLoader來解密後才能被JVM識別。
    3. 熱部署,同一個class文件通過不同的類載入器產生不同的class對象。兩個類完全相同不僅僅要是同一個class文件,還需要同一個類載入器、同一個JVM。

    代理模式 ,就是像上面圖片所展示那樣,當要載入一個類時,載入器會尋求其父類的幫助,讓父類嘗試去載入這個類。只有當父類失敗後,才會由自己載入。ClassLoader的loadClass()方法中體現。

    示例代碼:

    protected Class<?> loadClass(String name, boolean resolve)
          throws ClassNotFoundException
      {
          synchronized (getClassLoadingLock(name)) {
              Class<?> c = findLoadedClass(name);
              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
                  }
    
                  if (c == null) {
                      // If still not found, then invoke findClass in order
                      // 如果都沒有找到,則通過自定義實現的findClass去查找並載入
                      c = findClass(name);
    
                  }
              }
              if (resolve) {//是否需要在載入時進行解析
                  resolveClass(c);
              }
              return c;
          }
      }
    
    

    findClass(String name)的類似實現上面有示例,resolveClass()就是完成解析功能。

    URLClassLoader

    URLClassLoader是常用的ClassLoader類,其實現了findclass()介面,所以如果自定義時繼承URLClassLoader可以不用重寫findclass()。ExtClassLoader在代理模式中屬於Extension ClassLoader,而AppClassLoader屬於System ClassLoader。

    線程上下文載入器

    前面我們提到過BootStrap ClassLoader載入的是%JAVA_HOME%/lib下的核心庫文件,而CLASSPATH路徑下的庫由System ClassLoader載入。但在java語言中,存在這種現象,Java 提供了很多服務提供者介面(Service Provider Interface,SPI),允許第三方為這些介面提供實現,如JDBC、JCE、JNDI等。而SPI是Java的核心庫文件,由 BootStrap ClassLoader載入,第三方實現是放在CLASSPATH路徑下,由System ClassLoader載入。當BootStrap ClassLoader啟用時,需要載入其實現,但自己找不到,又因為代理模式的存在,無法委托System ClassLoader來載入,所以無法實現。

    Contex ClassLoader(線程上下文載入器)剛好可以解決這個問題。

    Contex ClassLoader(線程上下文載入器)是從 JDK 1.2 開始引入的。類 java.lang.Thread中的方法 getContextClassLoader()setContextClassLoader(ClassLoader cl)用來獲取和設置線程的上下文類載入器。如果沒有通過 setContextClassLoader(ClassLoader cl)方法進行設置的話,線程將繼承其父線程的上下文類載入器。Java 應用運行的初始線程的上下文類載入器是系統類載入器。線上程中運行的代碼可以通過此類載入器來載入類和資源。

    例子JDBC:

    public class DriverManager {
    	//省略......
        static {
            loadInitialDrivers();
            println("JDBC DriverManager initialized");
        }
    
     private static void loadInitialDrivers() {
         sun.misc.Providers()
         AccessController.doPrivileged(new PrivilegedAction<Void>() {
                public Void run() {
                    ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);
                  //省略不必要的代碼......
                }
            });
        }
    public class Driver extends com.mysql.cj.jdbc.Driver {
        public Driver() throws SQLException {
            super();
        }
    
        static {
          //省略
        }
    }
    public static <S> ServiceLoader<S> load(Class<S> service) {
    	 //通過線程上下文類載入器載入
          ClassLoader cl = Thread.currentThread().getContextClassLoader();
          return ServiceLoader.load(service, cl);
      }
    //調用
    String url = "jdbc:mysql://localhost:3342/cm-storylocker?characterEncoding=UTF-8";
    // 通過java庫獲取資料庫連接
    Connection conn = java.sql.DriverManager.getConnection(url, "root", "password");
    
    
    

    我們可以看到,當我們在使用JDBC的時候,會使用有DriverManager類,它的static代碼區引用的ServiceLoader類會完成JDBC實現類的載入。

    如何設計自己的ClassLoader

    給出例子,重寫文件系統載入器

    public class FileSystemClassLoader extends ClassLoader { 
     
       private String rootDir; 
     
       public FileSystemClassLoader(String rootDir) { 
           this.rootDir = rootDir; 
       } 
     
       protected Class<?> findClass(String name) throws ClassNotFoundException { 
           //根據規則獲取位元組流數組
           byte[] classData = getClassData(name); 
           if (classData == null) { 
               throw new ClassNotFoundException(); 
           } 
           else { 
               return defineClass(name, classData, 0, classData.length); 
           } 
       } 
     //設定自己的讀取規則
       private byte[] getClassData(String className) { 
           String path = classNameToPath(className); 
           try { 
               InputStream ins = new FileInputStream(path); 
               ByteArrayOutputStream baos = new ByteArrayOutputStream(); 
               int bufferSize = 4096; 
               byte[] buffer = new byte[bufferSize]; 
               int bytesNumRead = 0; 
               while ((bytesNumRead = ins.read(buffer)) != -1) { 
                   baos.write(buffer, 0, bytesNumRead); 
               } 
               return baos.toByteArray(); 
           } catch (IOException e) { 
               e.printStackTrace(); 
           } 
           return null; 
       } 
     
       private String classNameToPath(String className) { 
           return rootDir + File.separatorChar 
                   + className.replace('.', File.separatorChar) + ".class"; 
       } 
    }
    

    本文學習參考自

    [1] https://www.ibm.com/developerworks/cn/java/j-lo-classloader/index.html#minor1.1

    [2] https://blog.csdn.net/javazejian/article/details/73413292

    [3] https://juejin.im/post/5e1aaf626fb9a0301d11ac8e#heading-8


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

-Advertisement-
Play Games
更多相關文章
  • 正常我們寫一個左右兩列,左側一列放置圖片的html,如下所示: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> <style> .container{ background-col ...
  • 在vue項目中,為了減少首屏載入的時間,通常會開啟路由的懶載入。路由懶載入配合gizp確實能幫助我們大大的加快首屏的載入時間。 然而,路由懶載入會使得我們在第一次打開一個新頁面的時候,會有一個載入時間。如果在這個時候我們沒有一個提示的話,給人的感覺會是好像我點了頁面跳轉但是沒反應。所以,這個時候我們 ...
  • 先上一張咱們的原始圖 rs7.jpg 如下: 一、mask-image屬性 1.基本的png圖遮罩展示 我們使用的遮罩圖片 ad.png,如下: html: <img src="rs7.jpg" alt="" class="mask-img"> css: .mask-img { width: 300 ...
  • 最詳細的css3選擇器解析 選擇器是什麼? 比較官方的解釋:在 CSS 中,選擇器是一種模式,用於選擇需要添加樣式的元素。 最常見的 CSS 選擇器是元素選擇器。換句話說,文檔的元素就是最基本的選擇器。 看代碼,元素選擇器就是這個: h1作為一個元素標簽,是最基本的選擇器,這樣可以對h1標簽設置屬性 ...
  • 首先保證路徑正確,引入ttf字體,沒有生效 使用線上工具把ttf轉為更多常用的字體格式 網址:https://www.fontke.com/tool/convfont/ css里補全格式: 生效了 ...
  • 導讀 轉載自 "冪等性如何實現?深入瞭解一波!!!" 現在這個時代大家可能最關心的就是錢了,那麼有沒有想過你銀行轉賬給你沒有一次是轉多的,要麼失敗,要麼成功,為什麼不能失誤一下多轉一筆呢?醒醒吧年輕人,別做夢了,做銀行的能那麼傻x嗎? 今天我們就來談一談為什麼銀行轉賬不能多給我轉一筆?關乎到錢的問題 ...
  • 功能介紹:黑貓關鍵詞URL採集工具 Pro v1.0 批量關鍵詞自動搜索採集 自動去除垃圾二級泛解析功能變數名稱 可設置是否保存功能變數名稱或者url 聯繫客服QQ:944520563 ...
  • python線性數據結構 [TOC] 1 線性數據結構 本章要介紹的線性結構:list、tuple、string、bytes、bytearray。 線性表:是一種抽象的數學概念,是一組元素的序列的抽象,由有窮個元素組成(0個或任意個)。 線性表又可分為 順序表和鏈接表。 順序表:一組元素在記憶體中有序 ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...