Car.class載入到JVM中,稱為DNA後設資料模板,放在方法區。
虛擬機器器對Class檔案採用的是按需載入的方式:當需要使用該類時才會將它的Class檔案載入到記憶體生成Class位元組碼物件【方法區:Hotspot的元空間】
類的載入過程主要分為三個過程:載入、連結、初始化,其中連結又分為:驗證、準備、解析
(1)為靜態成員變數(不包括被final修飾的)分配記憶體,並設定該靜態成員變數的預設初始值。如果靜態成員變數
(2)被final修飾的靜態成員變數在準備階段會顯示的初始化值
(3)準備階段不會為普通成員變數賦值,因為靜態成員變數分配在方法區中,而普通成員變數是會隨著物件一起分配到堆記憶體中
public static int num=123;
注意:變數num在準備階段後的初始值為0而不是123
因為這個時候尚未開始執行任何方法,把123賦值給num是在初始化階段,在構造方法中進行的
public static final int num=123;
當final修飾的靜態成員變數,在準備階段初始化為123而非0,初始化時會把此語句和靜態程式碼塊會被合併到初始化構造方法中
public int num=1;
public final num=1;
普通成員變數不會再初始化階段賦預設值或賦指定值,因為普通成員變數會隨著物件存入堆記憶體中
JVM規範中將類的載入器分為兩類:啟動類載入器(Bootstrap ClassLoader)、自定義類載入器(User-Defined ClassLoader),其中所有派生於抽象類ClassLoader的類載入器都是自定義類載入器。
hotspot中類載入器分類:BootstrapClassLoader、ExtClassLoader、SystemClassLoader、使用者自定義類載入器,從前往後是一種包含關係,但不是繼承關係。
ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader()
為什麼要自定義類的載入器?
工作原理:如果一個類載入器收到了類載入請求,它並不會自己先去載入,而是委託給父類別載入器載入,如果父類別載入器還存在父載入器,則繼續向上委託,最終將類載入請求委託給啟動類載入器執行。如果父類別載入器可以完成類的載入,則成功返回,如果父類別載入器不能完成載入,則子載入器才會去嘗試載入
各載入器到各自負責的路徑下載入,不能載入該類,則子類到自己負責的路徑下載入。
舉例: 載入JDBC驅動時,系統類載入器將請求一層一層向上委託給啟動類載入器,但啟動類載入器、擴充套件類載入器都無法載入,最終又反向委派給系統類載入器載入。
在專案中將建立一個與String包名相同的包java.lang,並自定義String類:
public class String {
public static void main(String[] args) {
System.out.println("驗證雙親委派機制");
}
}
分析:
執行main方法時,應用程式類載入器收到載入java.lang包下String類的請求,一路向上委託至啟動類載入器,啟動類載入到java的核心類庫中載入了String類,而核心類庫中的String類沒有main方法,因此報錯:找不到main方法
上面這種保護核心API不被篡改的機制即:沙箱安全機制
(1)避免類的重複載入
(2)保護核心API被篡改(沙箱安全機制)
以上兩點均可以通過驗證雙親委派機制的例子證明
(1)類的完全限定名必須相同
(2)載入Class型別物件對應的Class檔案的ClassLoader範例物件必須相同(同一型別的同一個物件)
System.out.println(Demo.class.getClassLoader());
System.out.println(Thread.currentThread().getContextClassLoader());
System.out.println(ClassLoader.getSystemClassLoader());
public class Demo {
/**
* BootstrapClassLoader、SystemClassLoader、ExtClassLoader、使用者自定義類載入器,從前往後是一種包含關係,
* 但不是繼承關係
*/
public static void main(String[] args) throws ClassNotFoundException {
//獲取系統類載入器
ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
System.out.println(systemClassLoader);//sun.misc.Launcher$AppClassLoader@18b4aac2
//獲取擴充套件類載入器
ClassLoader extClassLoader = systemClassLoader.getParent();
System.out.println(extClassLoader);//sun.misc.Launcher$ExtClassLoader@725bef66
//獲取啟動類載入器
ClassLoader bootstrapClassLoader = extClassLoader.getParent();
System.out.println(bootstrapClassLoader);//null:啟動類載入器是用C或C++編寫的,無法獲取物件
/**
* 對比發現:我們自定義類的載入器和應用程式類載入器的地址是一樣的,因此自定義類使用的是應用類載入器
*/
System.out.println(Demo.class.getClassLoader());//sun.misc.Launcher$AppClassLoader@18b4aac2
//String類的載入器則是啟動類載入器,間接證明:Java的核心類庫是都是使用啟動類載入器載入
System.out.println(String.class.getClassLoader());//null
//獲取啟動類載入器的可以載入jar包的路徑
URL[] urLs = Launcher.getBootstrapClassPath().getURLs();
for (URL urL : urLs) {
System.out.println(urL);
}
//獲取擴充套件類載入器的可以載入jar包的路徑
String property = System.getProperty("java.ext.dirs");
System.out.println(property);
}
}