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
  • 移動開發(一):使用.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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...