虛擬機(jī)規(guī)范嚴(yán)格規(guī)定了有且只有5種情況必須立即對(duì)類進(jìn)行初始化(而加載、驗(yàn)證叁温、準(zhǔn)備自然需要在此之前開(kāi)始):
- 使用new關(guān)鍵字實(shí)例化對(duì)象的時(shí)候、讀取或設(shè)置一個(gè)類的靜態(tài)字段(被final修飾慨仿、已在編譯期把結(jié)果放入常量池的靜態(tài)字段除外)的時(shí)候所坯,以及調(diào)用一個(gè)類的靜態(tài)方法的時(shí)候。
- 使用java.lang.reflect包的方法對(duì)類進(jìn)行反射調(diào)用的時(shí)候锋八,如果類沒(méi)有進(jìn)行過(guò)初始化浙于,則需要先出發(fā)其初始化。
- 當(dāng)初始化一個(gè)類的時(shí)候挟纱,如果發(fā)現(xiàn)其父類還沒(méi)有進(jìn)行過(guò)初始化羞酗,則需要先觸發(fā)其父類的初始化。
- 當(dāng)虛擬機(jī)啟動(dòng)時(shí)紊服,用戶需要指定一個(gè)要執(zhí)行的主類(包含main()方法的那個(gè)類)檀轨,虛擬機(jī)會(huì)先初始化這個(gè)主類。
- 當(dāng)使用JDK1.7的動(dòng)態(tài)語(yǔ)言支持時(shí)欺嗤,如果一個(gè)java.lang.invoke.MethodHandle實(shí)力最后的解析結(jié)果REF_getStatic参萄、REF_putStatic、REF_invokeStatic的方法句柄煎饼,并且這個(gè)方法句柄所對(duì)應(yīng)的類沒(méi)有進(jìn)行過(guò)初始化讹挎,則需要先觸發(fā)其初始化。
哪些情況下不會(huì)對(duì)類進(jìn)行初始化:
- 通過(guò)子類引用父類的靜態(tài)字段吆玖,不會(huì)導(dǎo)致子類初始化筒溃。
public class Main {
public static void main(String[] args) throws Throwable {
System.out.println(SubClass.value);
}
}
class SuperClass{
static {
System.out.println("SuperClass init!");
}
public static int value=123;
}
class SubClass extends SuperClass{
static {
System.out.println("SubClass init");
}
}
//print:
//SuperClass init!
//123
- 通過(guò)數(shù)組定義來(lái)引用類,不會(huì)出發(fā)此類的初始化衰伯。
通過(guò)數(shù)組來(lái)引用的類铡羡,不會(huì)初始化此類,但是會(huì)初始化另一個(gè)由虛擬機(jī)自動(dòng)生成的意鲸,繼承于Object類的子類烦周,數(shù)組對(duì)象中的屬性和方法都實(shí)現(xiàn)在這個(gè)類里面尽爆。
public class Main {
public static void main(String[] args) throws Throwable {
SuperClass[] a=new SuperClass[10];
}
}
//print nothing
- 通過(guò)static final修飾的基本數(shù)據(jù)類型或String實(shí)例(或者稱為常量)在編譯階段會(huì)存入調(diào)用類的常量池中,本質(zhì)上并沒(méi)有直接引用到定義常量的類读慎,因此不會(huì)出發(fā)定義 常量的類的初始化漱贱。
public class Main {
public static void main(String[] args) throws Throwable {
System.out.println(ConstClass.HELLOWORLD);//print:hello world!
}
}
class ConstClass{
static {
System.out.println("ConstClass init!");
}
public static final String HELLOWORLD="hello world!";
}
字符串HELLOWORLD在編譯階段存入了調(diào)用類Main的常量池中,因此對(duì)字符串的引用被轉(zhuǎn)化為Main類對(duì)自身常量池的引用夭委。實(shí)際上Main的class文件中并沒(méi)有ConstClass類的符號(hào)引用入口幅狮,兩個(gè)類在編譯成class文件之后就不存在任何聯(lián)系了。
- 接口的加載過(guò)程和類的加載過(guò)程有些不同株灸。
接口不能用靜態(tài)語(yǔ)句崇摄,但編譯器仍然會(huì)為接口生成“<clint()>”類構(gòu)造器,用于初始化接口中所定義的成員變量(接口中的成員變量都是static final
)慌烧。
與類初始化的真正區(qū)別在于前面的初始化場(chǎng)景的第三種逐抑,當(dāng)一個(gè)類在初始化時(shí),要求其父類全部都已經(jīng)初始化過(guò)了屹蚊,但是接口初始化時(shí)厕氨,不要求其父類接口全部完成初始化,只有在真正使用到父類接口的時(shí)候(如引用接口中定義的常量)才會(huì)初始化
來(lái)自《深入理解java虛擬機(jī)》