JAVA學習之ClassLoader

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

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


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

更多相關文章
  • 正常我們寫一個左右兩列,左側一列放置圖片的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個或任意個)。 線性表又可分為 順序表和鏈接表。 順序表:一組元素在記憶體中有序 ...
一周排行
  • 上一篇文章(https://www.cnblogs.com/meowv/p/12943699.html)完成了項目的全局異常處理和日誌記錄。 在日誌記錄中使用的靜態方法有人指出寫法不是很優雅,遂優化一下上一篇中日誌記錄的方法,具體操作如下: 在.ToolKits層中新建擴展方法Log4NetExte ...
  • 先安裝幾個包 獲取地址如下https://www.nuget.org/packages/QRCoder/https://www.nuget.org/packages/SixLabors.Fonts/https://www.nuget.org/packages/SixLabors.ImageSharp ...
  • 0. 前言 前一篇我們詳細的介紹了SqlSugar的增刪改查,那些已經滿足我們在日常工程開發中的使用了。但是還有一點點在開發中並不常用,但是卻非常有用的方法。接下來讓我們一起來看看還有哪些有意思的內容。 1. 不同尋常的查詢 之前介紹了針對單個表的查詢,同樣也是相對簡單的查詢模式。雖然開發完全夠用, ...
  • 最新.net core 圖片合併生成二維碼合成圖片 圖片添加文字 先安裝幾個包 獲取地址如下 https://www.nuget.org/packages/QRCoder/ https://www.nuget.org/packages/SixLabors.Fonts/ https://www.nug ...
  • 前言 之前我寫過一篇關於 Blazor WebAssembly 的文章瀏覽器中的 .Net Core —— Blazor WebAssembly 初體驗,如今已經更新到 RC-1,與預覽版有著較大的差異,為此補充這篇文章。 正文 與預覽版的主要差異 1、這次的候選版修改了大部分包名和命名空間,因此無 ...
  • 藍牙設置相關界面,以下是通過C#方式打開的幾個方式,記錄一下 藍牙設置界面 1.控制面板命令bthprops.cpl 可以用控制面板 control+bthprops.cpl,也可以直接bthprops.cpl。更多的命令見:所有運行命令指令大全、CMD & CPL:快捷系統命令和控制面板命令 bt ...
  • 一:背景 1. 講故事 去年阿裡聚石塔上的所有isv簡訊通道全部對接阿裡通信,我們就做了對接改造,使用阿裡提供的.net sdk。 網址:https://help.aliyun.com/document_detail/114480.html 同事當時使用的是ons-.net v1.1.3版本,程式上 ...
  • 一.相關介紹 Dockerfile:關於Dockerfile的使用說明,我在文章《讓.NetCore程式跑在任何有docker的地方》中有說到,這裡不在贅述,需要的可以先看下,本文主要介紹Jenkinsfile結合dockerfile配合使用,自動構建.NetCore應用程式。 Jenkinsfil ...
  • 當用戶嚮應用程式發出請求時,伺服器將解析該請求,生成響應,然後將結果發送給客戶端。用戶可能會在伺服器處理請求的時候中止請求。就比如說用戶跳轉到另一個頁面中獲取說關閉頁面。在這種情況下,我們希望停止所有正在進行的工作,以浪費不必要的資源。例如我們可能要取消SQL請求、http調用請求、CPU密集型操作 ...
  • 在.NET中,我們可以通過Task.WhenAll用來等待多個任務。任務完成之後,我們可以使用await等待他們來獲取結果。 Task<int> task1 = Task.Run(() => 1); Task<string> task2 = Task.Run(() => "hello"); awai ...