深入探究Java類載入機制

一、概述

Java虛擬機(JVM)是運行Java程序的重要平台。在JVM中,類的載入、連接、初始化是Java程序運行的基礎。在Java中,類是按需載入的,也就是在程序首次使用類的時候才會被載入。Java採用的是雙親委派模型的類載入機制,這個機制對於Java的生態系統是非常重要的。本文將從多個方面闡述Java類載入機制。

二、類的生命周期

在了解Java的類載入機制之前,需要了解類的生命周期:

  1. 載入:從文件系統、網路等來源讀取二進位數據,並且創建出Class對象。
  2. 連接:包括驗證、準備和解析三個階段。
  3. 初始化:為類的靜態變數賦初值,執行<clinit>方法。
  4. 使用:類被調用。
  5. 卸載:類不再被引用,被GC回收。

三、類載入機制

1. 雙親委派模型

Java採用的是雙親委派模型的類載入機制,這個機制對Java的生態系統是非常重要的。它是從Java1.2開始加入的,嚴格來說應該叫做雙親委派委派模型。簡單來說,就是當一個類載入器接收到一個類載入請求後,它會將請求轉發給它的父類載入器,直到最頂層的啟動類載入器。

    protected Class<?> loadClass(String name, boolean resolve)
            throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // 首先從緩存中查找已經被載入過的類
            Class<?> c = findLoadedClass(name);
            // 如果沒有被載入,再讓父親去試一下
            if (c == null) {
                try {
                    if (parent != null) {
                        c = parent.loadClass(name, false);
                    } else {
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException ignored) {}
                // 如果父親都找不到,最後就由自己去載入
                if (c == null) {
                    c = findClass(name);
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }

2. 雙親委派的好處

藉助於雙親委派機制,Java程序可以避免類的重複載入。這種機制從根本上保證了Java程序的穩定性和安全性。例如,在使用微服務和大型框架的情況下,如果每個服務請求使用不同的類載入器,就會造成類重複載入,從而會導致內存使用量過高、版本不一致等問題。通過引入雙親委派機制,可以將類載入的責任傳遞給父類載入器,避免了類重複載入問題。

3. 破壞雙親委派模型

雖然雙親委派模型時Java的默認類載入機制,但是它並不是萬能的。有時候,我們會遇到一些特殊場景,需要打破這個機制。Java提供了兩種方式來打破雙親委派機制:

  1. 線程上下文類載入器
  2. 為了解決上述問題,Java引入了線程上下文類載入器的概念。它是從Java1.2引入的。當默認的類載入器無法滿足某個類載入請求時,可以通過線程上下文類載入器來載入。可以通過Thread.currentThread().setContextClassLoader()方法來設置線程上下文類載入器。

        public static void main(String[] args) {
            Thread.currentThread().setContextClassLoader(new MyClassLoader());
            ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
            Class<?> clazz = classLoader.loadClass("com.example.MyClass");
        }
    
  3. 自定義類載入器
  4. 通過自定義類載入器,可以打破雙親委派模型,從而實現更靈活的類載入機制。自定義類載入器需要繼承java.lang.ClassLoader類,並重寫其中的findClass()方法。調用自定義類載入器的代碼應該在類和類載入器分離的情況下,通過newInstance()來創建新的類對象。

        public class MyClassLoader extends ClassLoader {
            @Override
            protected Class<?> findClass(String name) {
                // 從指定的磁碟路徑載入類
                byte[] bytes = loadClassBytes(name);
                return defineClass(name, bytes, 0, bytes.length);
            }
        }
    

四、常見問題

1. 類被重複載入了怎麼辦?

我們可以通過JVM參數-verbose:class來查看類的載入情況。如果出現類重複載入的情況,可以使用強制主動載入的方式來防止重複載入。例如,在使用spring-boot-devtools的場景下,可以在build.gradle文件中添加以下代碼:

plugins {
    id 'org.springframework.boot' version '2.3.1.RELEASE' apply false
}
 
ext['spring-boot'] = ["org.springframework.boot:spring-boot-devtools"]
dependencies {
    runtimeOnly "${spring-boot}"
}
 
configurations {
    developmentOnly
    runtimeClasspath {
        extendsFrom developmentOnly
    }
}
 
task developmentJar(type: Jar) {
    classifier = 'development'
    from sourceSets.main.output
    appendix = "-${version}-SNAPSHOT"
}
 
artifacts {
    developmentOnly developmentJar
}
 
bootJar {
    requiresUnpack '**/spring-boot-devtools-*.jar'
}
 
task devtools(additionalResourceDirs: sourceSets.main.output.resourcesDir) {
    doLast {
        files(additionalResourceDirs).visit {
            def file = it.getFile();
            if (file.path.endsWith('.class')) {
                Class.forName(file.path.split('main\\\\classes\\\\', 2)[1].replaceAll('\\\\', '.').replaceAll('.class', ''))
            }
        }
    }
}
 
tasks.compileJava {
    finalizedBy devtools
}

2. 打破雙親委派模型會帶來什麼問題?

在某些情況下,打破雙親委派機制可能會導致類重複載入、版本不一致等問題。例如:Maven的classpath中存在多個版本的同一依賴包時,使用默認的雙親委派模型無法載入到正確版本的類,這時就可以通過線程上下文類載入器來載入。但需要注意的是,過多地使用自定義類載入器會降低Java程序的性能。

五、總結

Java的類載入機制是Java程序運行的基礎,其中雙親委派模型是其重要的機制之一。通過對雙親委派模型的理解和應用,可以保證Java程序的穩定性和安全性。當然,在必要的情況下,也可以通過線程上下文類載入器或者自定義類載入器來打破默認的類載入機制,實現更靈活的類載入方式。

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-tw/n/156901.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2024-11-18 01:58
下一篇 2024-11-18 01:58

相關推薦

  • java client.getacsresponse 編譯報錯解決方法

    java client.getacsresponse 編譯報錯是Java編程過程中常見的錯誤,常見的原因是代碼的語法錯誤、類庫依賴問題和編譯環境的配置問題。下面將從多個方面進行分析…

    編程 2025-04-29
  • Java JsonPath 效率優化指南

    本篇文章將深入探討Java JsonPath的效率問題,並提供一些優化方案。 一、JsonPath 簡介 JsonPath是一個可用於從JSON數據中獲取信息的庫。它提供了一種DS…

    編程 2025-04-29
  • Java騰訊雲音視頻對接

    本文旨在從多個方面詳細闡述Java騰訊雲音視頻對接,提供完整的代碼示例。 一、騰訊雲音視頻介紹 騰訊雲音視頻服務(Cloud Tencent Real-Time Communica…

    編程 2025-04-29
  • QML 動態載入實踐

    探討 QML 框架下動態載入實現的方法和技巧。 一、實現動態載入的方法 QML 支持從 JavaScript 中動態指定需要載入的 QML 組件,並放置到運行時指定的位置。這種技術…

    編程 2025-04-29
  • Java Bean載入過程

    Java Bean載入過程涉及到類載入器、反射機制和Java虛擬機的執行過程。在本文中,將從這三個方面詳細闡述Java Bean載入的過程。 一、類載入器 類載入器是Java虛擬機…

    編程 2025-04-29
  • Java Milvus SearchParam withoutFields用法介紹

    本文將詳細介紹Java Milvus SearchParam withoutFields的相關知識和用法。 一、什麼是Java Milvus SearchParam without…

    編程 2025-04-29
  • Java 8中某一周的周一

    Java 8是Java語言中的一個版本,於2014年3月18日發布。本文將從多個方面對Java 8中某一周的周一進行詳細的闡述。 一、數組處理 Java 8新特性之一是Stream…

    編程 2025-04-29
  • Java判斷字元串是否存在多個

    本文將從以下幾個方面詳細闡述如何使用Java判斷一個字元串中是否存在多個指定字元: 一、字元串遍歷 字元串是Java編程中非常重要的一種數據類型。要判斷字元串中是否存在多個指定字元…

    編程 2025-04-29
  • VSCode為什麼無法運行Java

    解答:VSCode無法運行Java是因為默認情況下,VSCode並沒有集成Java運行環境,需要手動添加Java運行環境或安裝相關插件才能實現Java代碼的編寫、調試和運行。 一、…

    編程 2025-04-29
  • Java任務下發回滾系統的設計與實現

    本文將介紹一個Java任務下發回滾系統的設計與實現。該系統可以用於執行複雜的任務,包括可回滾的任務,及時恢復任務失敗前的狀態。系統使用Java語言進行開發,可以支持多種類型的任務。…

    編程 2025-04-29

發表回復

登錄後才能評論