一、概述
Java虛擬機(JVM)是運行Java程序的重要平台。在JVM中,類的載入、連接、初始化是Java程序運行的基礎。在Java中,類是按需載入的,也就是在程序首次使用類的時候才會被載入。Java採用的是雙親委派模型的類載入機制,這個機制對於Java的生態系統是非常重要的。本文將從多個方面闡述Java類載入機制。
二、類的生命周期
在了解Java的類載入機制之前,需要了解類的生命周期:
- 載入:從文件系統、網路等來源讀取二進位數據,並且創建出Class對象。
- 連接:包括驗證、準備和解析三個階段。
- 初始化:為類的靜態變數賦初值,執行
<clinit>
方法。 - 使用:類被調用。
- 卸載:類不再被引用,被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提供了兩種方式來打破雙親委派機制:
- 線程上下文類載入器
- 自定義類載入器
為了解決上述問題,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"); }
通過自定義類載入器,可以打破雙親委派模型,從而實現更靈活的類載入機制。自定義類載入器需要繼承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