一、類的初始化
對於類的初始化:類的初始化一般只初始化一次,類的初始化主要是初始化靜態成員變量。
類的編譯決定了類的初始化過程。
編譯器生成的class文件主要對定義在源文件中的類進行了如下的更改:
1) 先按照靜態成員變量的定義順序在類內部聲明成員變量。
2) 再按照原java類中對成員變量的初始化順序進行初始化。
一個java類和編譯後的class對應的轉換如下:
源文件:
代碼如下:
public class Person{
public static String name="張三";
public static int age;
static{
age=20;
System.out.println("初始化age");
}
public static String address;
static{
address="北京市";
age=34;
}
public static void main(String[] args) {
System.out.println(name);
System.out.println(age);
System.out.println(address);
}
}
當java源代碼轉換成一個class文件後,其轉換成類似下面的代碼:
代碼如下:
public class Person{
public static String name;
public static int age;
public static String address;
static{
name="張三";
age=20;
System.out.println("初始化age");
address="北京市";
age=34;
}
public static void main(String[] args) {
System.out.println(name);
System.out.println(age);
System.out.println(address);
}
}
初始化順序依據轉換後對應的class類成員變量的初始化順序依次執行,所以所有的靜態成員變量都是先聲明,後執行賦值的,而且賦值的順序也是依照源代碼對靜態成員變量初始化的順序進行的,注意:定義一個成員變量並直接初始化與在靜態代碼塊中進行初始化是等價的,都是依據它們在源代碼中定義的順序進行的。
二、對象的生成
對於對象的生成其初始化過程與類的初始化過程類似,但會增加構造函數階段,源代碼如下:
代碼如下:
public class Person{
{
name="李四";
age=56;
System.out.println("初始化age");
address="上海";
}
public String name="張三";
public int age=29;
public String address="北京市";
public Person(){
name="趙六";
age=23;
address="上海市";
}
}
編譯器轉換成class文件後,會轉換成類似下面的代碼:
代碼如下:
public class Person{
public String name;
public int age;
public String address;
public Person(){
name="李四";
age=56;
System.out.println("初始化age");
address="上海";
name="張三";
age=29;
address="北京市";
name="趙六";
age=23;
address="上海市";
}
}
可以看到,對於類中對成員變量的初始化和代碼塊中的代碼全部都挪到了構造函數中,並且是按照java源文件的初始化順序依次對成員變量進行初始化的,而原構造函數中的代碼則移到了構造函數的最後執行。本人以前一直對類初始化過程一直沒有一個深刻的理解,就是搞不清到底是怎麼初始化的,只能按照書上所說的記住了初始化順序,但是過一段時間有給忘了,所以這次總算給弄清楚了,還是按照一個模型來解釋初始化機制比較好啊,不用再背了,只有理解了才能不易忘