詳解Java 類的加載、連接和初始化
系統(tǒng)可能在第一次使用某個(gè)類時(shí)加載該類,也可能采用預(yù)加載機(jī)制來(lái)加載某個(gè)類。本節(jié)將會(huì)詳細(xì)介紹類加載、連接和初始化過(guò)程中的每個(gè)細(xì)節(jié)。
JVM 和類
當(dāng)調(diào)用 java 命令運(yùn)行某個(gè) Java 程序時(shí),該命令將會(huì)啟動(dòng)一個(gè) Java 虛擬機(jī)進(jìn)程,不管該 Java 程序有多么復(fù)雜,該程序啟動(dòng)了多少個(gè)線程,它們都處于該 Java 虛擬機(jī)進(jìn)程里。正如前面介紹的,同一個(gè) JVM 的所有線程、所有變量都處于同一個(gè)進(jìn)程里,它們都使用該 JVM 進(jìn)程的內(nèi)存區(qū)。當(dāng)系統(tǒng)出現(xiàn)以下幾種情況時(shí),JVM 進(jìn)程將被終止。
- 程序運(yùn)行到最后正常結(jié)束。
- 程序運(yùn)行到使用 system.exit() 或 Runtime.getRuntime().exit() 代碼處結(jié)束程序。
- 程序執(zhí)行過(guò)程中遇到未捕獲的異?;蝈e(cuò)誤而結(jié)束。
- 程序所在平臺(tái)強(qiáng)制結(jié)束了 JVM 進(jìn)程。
從上面的介紹可以看出,當(dāng) Java 程序運(yùn)行結(jié)束時(shí),JVM 進(jìn)程結(jié)束,該進(jìn)程在內(nèi)存中的狀態(tài)將會(huì)丟失。下面以類的類變量來(lái)說(shuō)明這個(gè)問(wèn)題。下面程序先定義了一個(gè)包含類變量的類。
public class A {
// 定義該類的類變量
public static int a = 6;
}
上面程序中的粗體字代碼定義了一個(gè)類變量a,接下來(lái)定義一個(gè)類創(chuàng)建A類的實(shí)例,并訪問(wèn)A對(duì)象的類變量a。
public class ATest1 {
public static void main(String[] args) {
// 創(chuàng)建A類的實(shí)例
A a = new A();
// 讓a實(shí)例的類變量a的值自加
a.a++;
System.out.println(a.a);
}
}
下面程序也創(chuàng)建A對(duì)象,并訪問(wèn)其類變量a的值。
public class ATest2 {
public static void main(String[] args) {
// 創(chuàng)建A類的實(shí)例
A b = new A();
// 輸出b實(shí)例的類變量a的值
System.out.println(b.a);
}
}
在 ATest1.java 程序中創(chuàng)建了A類的實(shí)例,并讓該實(shí)例的類變量a的值自加,程序輸出該實(shí)例的類變量a的值將看到7,相信讀者對(duì)這個(gè)答案沒(méi)有疑問(wèn)。關(guān)鍵是運(yùn)行第二個(gè)程序 ATest2 時(shí),程序再次創(chuàng)建了A對(duì)象,并輸出A對(duì)象類變量的a的值,此時(shí)a的值是多少呢?結(jié)果依然是6,并不是7。這是因?yàn)檫\(yùn)行 ATest1 和 ATest2 是兩次運(yùn)行 JVM 進(jìn)程,第一次運(yùn)行 JVM 結(jié)束后,它對(duì)A類所做的修改將全部丟失——第二次運(yùn)行 JVM 時(shí)將再次初始化A類。
注意:兩次運(yùn)行 Java 程序處于兩個(gè)不同的 JVM 進(jìn)程中,兩個(gè) JVM 之間并不會(huì)共享數(shù)據(jù)。
類的加載
當(dāng)程序主動(dòng)使用某個(gè)類時(shí),如果該類還未被加載到內(nèi)存中,則系統(tǒng)會(huì)通過(guò)加載、連接、初始化三個(gè)步驟來(lái)對(duì)該類進(jìn)行初始化。如果沒(méi)有意外,JVM 將會(huì)連續(xù)完成這三個(gè)步驟,所以有時(shí)也把這三個(gè)步驟統(tǒng)稱為類加載或類初始化。
類加載指的是將類的 class 文件讀入內(nèi)存,并為之創(chuàng)建一個(gè) java.lang.Class 對(duì)象,也就是說(shuō),當(dāng)程序中使用任何類時(shí),系統(tǒng)都會(huì)為之建立一個(gè) java.lang.Class 對(duì)象。
提示:前面介紹面向?qū)ο髸r(shí)提到:類是某一類對(duì)象的抽象,類是概念層次的東西.但不知道讀者有沒(méi)有想過(guò):類也是一種對(duì)象就像平常說(shuō)概念主要用于定義、描述其他事物,但概念本身也是一種事物,那么概念本身也需要被描述———這有點(diǎn)像一個(gè)哲學(xué)命題,但事實(shí)就是這樣,每個(gè)類是一批具有相同特征的對(duì)象的抽象(或者說(shuō)概念),而系統(tǒng)中所有的類實(shí)際上也是實(shí)例,它們都是 java.lang.Class 的實(shí)例。
類的加載由類加載器完成,類加載器通常由 JVM 提供,這些類加載器也是前面所有程序運(yùn)行的基礎(chǔ),JVM 提供的這些類加載器通常被稱為系統(tǒng)類加載器。除此之外,開發(fā)者可以通過(guò)繼承 ClassLoader 基類來(lái)創(chuàng)建自己的類加載器。
通過(guò)使用不同的類加載器,可以從不同來(lái)源加載類的二進(jìn)制數(shù)據(jù),通常有如下幾種來(lái)源。
- 從本地文件系統(tǒng)加載 class 文件,這是前面絕大部分示例程序的類加載方式。
- 從 JAR 包加載 class 文件,這種方式也是很常見的,前面介紹 JDBC 編程時(shí)用到的數(shù)據(jù)庫(kù)驅(qū)動(dòng)類就放在 JAR 文件中,JVM 可以從 JAR 文件中直接加載該 class 文件。
- 通過(guò)網(wǎng)絡(luò)加載 class 文件。
- 把一個(gè) Java 源文件動(dòng)態(tài)編譯,并執(zhí)行加載。
類加載器通常無(wú)須等到“首次使用”該類時(shí)才加載該類,Java 虛擬機(jī)規(guī)范允許系統(tǒng)預(yù)先加載某些類。
類的連接
當(dāng)類被加載之后,系統(tǒng)為之生成一個(gè)對(duì)應(yīng)的 Class 對(duì)象,接著將會(huì)進(jìn)入連接階段,連接階段負(fù)責(zé)把類的二進(jìn)制數(shù)據(jù)合并到 JRE 中。類連接又可分為如下三個(gè)階段。
(1)驗(yàn)證:驗(yàn)證階段用于檢驗(yàn)被加載的類是否有正確的內(nèi)部結(jié)構(gòu),并和其他類協(xié)調(diào)一致。
(2)準(zhǔn)備:類準(zhǔn)備階段則負(fù)責(zé)為類的類變量分配內(nèi)存,并設(shè)置默認(rèn)初始值。
(3)解析:將類的二進(jìn)制數(shù)據(jù)中的符號(hào)引用替換成直接引用。
類的初始化
在類的初始化階段,虛擬機(jī)負(fù)責(zé)對(duì)類進(jìn)行初始化,主要就是對(duì)類變量進(jìn)行初始化。在 Java 類中對(duì)類變量指定初始值有兩種方式:
①聲明類變量時(shí)指定初始值;
②使用靜態(tài)初始化塊為類變量指定初始值。例如下面代碼片段。
public class Test {
// 聲明變量a時(shí)指定初始值
static int a = 5;
static int b = 9; // ①
static int c;
static {
// 使用靜態(tài)初始化塊為變量b指定出初始值
b = 6;
System.out.println("----------");
}
public static void main(String[] args) {
System.out.println(Test.b);
}
}
對(duì)于上面代碼,程序?yàn)轭愖兞縜、b都顯式指定了初始值,所以這兩個(gè)類變量的值分別為5、6,但類變量c則沒(méi)有指定初始值,它將采用默認(rèn)初始值0。
聲明變量時(shí)指定初始值,靜態(tài)初始化塊都將被當(dāng)成類的初始化語(yǔ)句,JVM 會(huì)按這些語(yǔ)句在程序中的排列順序依次執(zhí)行它們,例如下面的類。
public class Test {
static {
// 使用靜態(tài)初始化塊為變量b指定出初始值
b = 6;
System.out.println("----------");
}
// 聲明變量a時(shí)指定初始值
static int a = 5;
static int b = 9; // ①
static int c;
public static void main(String[] args) {
System.out.println(Test.b);
}
}
上面代碼先在靜態(tài)初始化塊中為b變量賦值,此時(shí)類變量b的值為6;接著程序向下執(zhí)行,執(zhí)行到①號(hào)代碼處,這行代碼也屬于該類的初始化語(yǔ)句,所以程序再次為類變量b賦值。也就是說(shuō),當(dāng) Test 類初始化結(jié)束后,該類的類變量b的值為9。
JVM 初始化一個(gè)類包含如下幾個(gè)步驟。
①假如這個(gè)類還沒(méi)有被加載和連接,則程序先加載并連接該類。
②假如該類的直接父類還沒(méi)有被初始化,則先初始化其直接父類。
③假如類中有初始化語(yǔ)句,則系統(tǒng)依次執(zhí)行這些初始化語(yǔ)句。
當(dāng)執(zhí)行第2個(gè)步驟時(shí),系統(tǒng)對(duì)直接父類的初始化步驟也遵循此步驟1、3;如果該直接父類又有直接父類,則系統(tǒng)再次重復(fù)這三個(gè)步驟來(lái)先初始化這個(gè)父類......依此類推,所以 JVM 最先初始化的總是 java.lang.Object 類。當(dāng)程序主動(dòng)使用任何一個(gè)類時(shí),系統(tǒng)會(huì)保證該類以及所有父類(包括直接父類和間接父類〕都會(huì)被初始化。
類初始化的時(shí)機(jī)
當(dāng) Java 程序首次通過(guò)下面6種方式來(lái)使用某個(gè)類或接口時(shí),系統(tǒng)就會(huì)初始化該類或接口。
- 創(chuàng)建類的實(shí)例。為某個(gè)類創(chuàng)建實(shí)例的方式包括:使用 new 操作符來(lái)創(chuàng)建實(shí)例,通過(guò)反射來(lái)創(chuàng)建實(shí)例,通過(guò)反序列化的方式來(lái)創(chuàng)建實(shí)例。
- 調(diào)用某個(gè)類的類方法(靜態(tài)方法)。
- 訪問(wèn)某個(gè)類或接口的類變量,或?yàn)樵擃愖兞抠x值。
- 使用反射方式來(lái)強(qiáng)制創(chuàng)建某個(gè)類或接口對(duì)應(yīng)的 java.lang.Class 對(duì)象。例如代碼:Class.forName("Person"),如果系統(tǒng)還未初始化 Person 類,則這行代碼將會(huì)導(dǎo)致該 Person 類被初始化,并返回 Person 類對(duì)應(yīng)的 java.lang.Class 對(duì)象。
- 初始化某個(gè)類的子類。當(dāng)初始化某個(gè)類的子類時(shí),該子類的所有父類都會(huì)被初始化。
- 直接使用 java.exe 命令來(lái)運(yùn)行某個(gè)主類。當(dāng)運(yùn)行某個(gè)主類時(shí),程序會(huì)先初始化該主類。
除此之外,下面的幾種情形需要特別指出。
對(duì)于一個(gè) final 型的類變量,如果該類變量的值在編譯時(shí)就可以確定下來(lái),那么這個(gè)類變量相當(dāng)于“宏變量”。Java 編譯器會(huì)在編譯時(shí)直接把這個(gè)類變量出現(xiàn)的地方替換成它的值,因此即使程序使用該靜態(tài)類變量,也不會(huì)導(dǎo)致該類的初始化。例如下面示例程序的結(jié)果。
class MyTest {
static {
System.out.println("靜態(tài)初始化塊...");
}
// 使用一個(gè)字符串直接量為static final的類變量賦值
static final String compileConstant = "瘋狂Java講義";
}
public class CompileConstantTest {
public static void main(String[] args) {
// 訪問(wèn)、輸出MyTest中的compileConstant類變量
System.out.println(MyTest.compileConstant); // ①
}
}
上面程序的 MyTest 類中有一個(gè) compileConstant 的類變量,該類變量使用了 final 修飾,而且它的值可以在編譯時(shí)確定下來(lái),因此 compileConstant 會(huì)被當(dāng)成“宏變量”處理。程序中所有使用 compileConstant 的地方都會(huì)在編譯時(shí)被直接替換成它的值——也就是說(shuō),上面程序中①處的粗體字代碼在編譯時(shí)就會(huì)被替換成“瘋狂Java講義”,所以①行代碼不會(huì)導(dǎo)致初始化 MyTest 類。
提示:當(dāng)某個(gè)類變量(也叫靜態(tài)變量)使用了 final 修飾,而且它的值可以在編譯時(shí)就確定下來(lái),那么程序其他地方使用該類變量時(shí),實(shí)際上并沒(méi)有使用該類變量,而是相當(dāng)于使用常量。
反之,如果 final 修飾的類變量的值不能在編譯時(shí)確定下來(lái).則必須等到運(yùn)行時(shí)才可以確定該類變量的值,如果通過(guò)該類來(lái)訪問(wèn)它的類變量,則會(huì)導(dǎo)致該類被初始化。例如將上面程序中定義compileConstant 的代碼改為如下:
//采用系統(tǒng)當(dāng)前時(shí)間為 static final 類變量賦值 static final String compileConstant = System.currentTimeMiIlis() + "";
因?yàn)樯厦娑x的 compileConstant 類變量的值必須在運(yùn)行時(shí)才可以確定,所以①處的粗體字代碼必須保留為對(duì) MyTest 類的類變量的引用,這行代碼就變成了使用 MyTest 的類變量,這將導(dǎo)致 MyTest 類被初始化。
當(dāng)使用 ClassLoader 類的 loadClass() 方法來(lái)加載某個(gè)類時(shí),該方法只是加載該類,并不會(huì)執(zhí)行該類的初始化。使用 Class 的 forName() 靜態(tài)方法才會(huì)導(dǎo)致強(qiáng)制初始化該類。例如如下代碼。
package com.jwen.chapter18_1;
class Tester {
static {
System.out.println("Tester類的靜態(tài)初始化塊...");
}
}
public class ClassLoaderTest {
public static void main(String[] args) throws ClassNotFoundException {
ClassLoader cl = ClassLoader.getSystemClassLoader();
// 下面語(yǔ)句僅僅是加載Tester類
cl.loadClass("com.jwen.chapter18_1.Tester");
System.out.println("系統(tǒng)加載Tester類");
// 下面語(yǔ)句才會(huì)初始化Tester類
Class.forName("com.jwen.chapter18_1.Tester");
}
}
上面程序中的兩行粗體字代碼都用到了 Tester 類,但第一行粗體字代碼只是加載 Tester 類,并不會(huì)初始化 Tester 類。運(yùn)行上面程序,會(huì)看到如下運(yùn)行結(jié)果:
系統(tǒng)加載Tester類
Tester類的靜態(tài)初始化塊...
從上面運(yùn)行結(jié)果可以看出,必須等到執(zhí)行 Class.forName("Tester") 時(shí)才完成對(duì) Tester 類的初始化。
以上就是詳解Java 類的加載、連接和初始化的詳細(xì)內(nèi)容,更多關(guān)于Java 類的加載、連接和初始化的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
JAVA實(shí)現(xiàn)空間索引編碼——GeoHash的示例
本篇文章主要介紹了JAVA實(shí)現(xiàn)空間索引編碼——GeoHash的示例,如何從眾多的位置信息中查找到離自己最近的位置,有興趣的朋友可以了解一下2016-10-10
詳解使用JRebel插件實(shí)現(xiàn)SpringBoot應(yīng)用代碼熱加載
這篇文章主要介紹了詳解使用JRebel插件實(shí)現(xiàn)SpringBoot應(yīng)用代碼熱加載,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10
netty對(duì)proxy protocol代理協(xié)議的支持詳解
這篇文章主要為大家介紹了netty對(duì)proxy protoco代理協(xié)議的支持詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
從零開始學(xué)Java之關(guān)系運(yùn)算符
今天帶大家復(fù)習(xí)Java關(guān)系運(yùn)算符,文中對(duì)Java運(yùn)算符相關(guān)知識(shí)作了詳細(xì)總結(jié),對(duì)正在學(xué)習(xí)java基礎(chǔ)的小伙伴們很有幫助,需要的朋友可以參考下2021-08-08
Java Fluent Mybatis 項(xiàng)目工程化與常規(guī)操作詳解流程篇 下
Java中常用的ORM框架主要是mybatis, hibernate, JPA等框架。國(guó)內(nèi)又以Mybatis用的多,基于mybatis上的增強(qiáng)框架,又有mybatis plus和TK mybatis等。今天我們介紹一個(gè)新的mybatis增強(qiáng)框架 fluent mybatis關(guān)于項(xiàng)目工程化與常規(guī)操作流程2021-10-10
javacv視頻抽幀的實(shí)現(xiàn)過(guò)程詳解(附代碼)
這篇文章主要介紹了javacv視頻抽幀的實(shí)現(xiàn)過(guò)程詳解(附代碼),視頻抽幀可以做一些處理,比如水印,去水印等操作,然后再合成視頻,需要的朋友可以參考下2019-07-07
Idea調(diào)用WebService的關(guān)鍵步驟和注意事項(xiàng)
這篇文章主要介紹了如何在Idea中調(diào)用WebService,包括理解WebService的基本概念、獲取WSDL文件、閱讀和理解WSDL文件、選擇對(duì)接測(cè)試工具或方式、發(fā)送請(qǐng)求和接收響應(yīng)、處理響應(yīng)結(jié)果以及錯(cuò)誤處理,需要的朋友可以參考下2025-01-01

