類的初始化:在初始化階段,java虛擬機執行類的初始化語句,爲類的靜態變量賦予初始值。在程序中,靜態變量初始化有兩種途徑:
1、類的初始化:在初始化階段,java虛擬機執行類的初始化語句,爲類的靜態變量賦予初始值。在程序中,靜態變量初始化有兩種途徑:
(1)在靜態變量的聲明處進行初始化;
(2)在靜態代碼快中進行初始化。例如在以下代碼中,靜態變量a和b都被顯示初始化,而靜態變量c沒有被顯示初始化,它將保持默認值爲0;
public class ClassLoaderTest {
public static void main(String[] args) {
Singleton singleton = Singleton.getInstance();
System.out.println("counter1= " + singleton.counter1);
System.out.println("counter2= " + singleton.counter2);
}
}
/**
* 程序是從上向下順序執行 new Singleton()時,counter1,counter2初始值均爲0
* 在通過構造方法Singleton(),均加1.則返回的值counter1,counter2均爲1
* 然後再程序在繼續向下執行,由於counter1沒有顯示初始化,則值還是爲1 但是counter2經過顯示初始化後,其值爲0
*
* @author zhaojw_420
*/
class Singleton {
private static Singleton singleton = new Singleton();
public static int counter1;
public static int counter2 = 0;
private Singleton() {
counter1++;
counter2++;
}
public static Singleton getInstance() {
return singleton;
}
}
public class ClassLoaderTest {
public static void main(String[] args) {
System.err.println(Test1.X);
}
}
/**
* 當X=6/3時,編譯時即可算出X=2,即編譯時常量,即不需要運行類,所以不打印靜態代碼塊中的內容 當X=new
* Random().nextInt(100)時,編譯時不能算出X的值,只有運行程序才知道,所以打印結果爲:FinalTest static final 2
*
* @author zhaojw_420
*
*/
class Test1 {
//public static final int X=6/3;//打印結果: 2
public static final int X=new Random().nextInt(100);//打印結果爲FinalTest static final 2
static{
System.err.println("FinalTest static final");
}
}
類的初始化時機:當java虛擬機初始化一個類時,要求他的所有父類都已經被初始化,但是這條跪着並不適用於接口。
(1)在初始化一個類時,並不先初始化它所實現的接口
(2)在初始化一個接口時,並不會先初始化他的父接口。
因此,一個父接口並不會因爲他的子接口或者實現類的初始化而初始化。只有當程序首次使用特定接口的靜態變量時,纔會導致該接口的初始化。
public class ClassLoaderTest {
static {
System.err.println("Test4 static block");
}
/**運行結果
* Test4 static block
* Parent static block
* Child static block
* 4
* @author zhaojw_420
*
*/
public static void main(String[] args) {
System.err.println(Child.b);
}
}
class Parent{
public static final int a=3;
static{
System.err.println("Parent static block");
}
}
class Child extends Parent{
public static final int b=4;
static{
System.err.println("Child static block");
}
}