探究Java初始化的過程
最近又在翻《thinking in java》,這本書,怎么說呢,以前學(xué)java的時候,老師就沒有把它作為教材,但是我偏偏只買了這本書,一直收獲很大。好了,言歸正傳,結(jié)合自己的偶然遇到的一個在構(gòu)造函數(shù)中調(diào)多態(tài)方法引起的思考,講述一下java的初始化到底是怎樣的一個過程。
所謂初始化,當(dāng)然也就指的是變量。變量可以是內(nèi)置的變量或者我們創(chuàng)建的類的對象。
有人說,本來初始化本來就是一件很簡單的事情,的確,但是java作為一門面向?qū)ο笳Z言,由于具有繼承、多態(tài),靜態(tài)、動態(tài)綁定等多種特性,所以其初始化的情景可謂是五花八門?,F(xiàn)在就一步一步的分析其初始化過程。下面就是幾個小原則。優(yōu)先級依次遞減。
1、靜態(tài)塊優(yōu)先
程序首先會執(zhí)行靜態(tài)塊的內(nèi)容,這也就有了不寫main方法就跑hello world的小故事,相信說到這里,大家就有了思路。我們都知道靜態(tài)類型是和類綁定的而不是和具體實例對象綁定。也就是說,引用一個靜態(tài)變量的方式往往是MyClass.xxx.這個特點決定了其在編譯的階段就已經(jīng)分配好了固定的空間。
2、父類優(yōu)先
由于繼承的特性,當(dāng)導(dǎo)出類(子類)的對象被創(chuàng)建的時候,程序?qū)⑾蛏献匪莸阶畛醯母割悾瑘?zhí)行其初始化的操作。然后一次向下調(diào)用子類的構(gòu)造函數(shù)。按照這個思路,那么每一個實例***個要初始化的必定是Object類了。
3、成員變量優(yōu)先
一定要注意,成員變量按照其聲明的順序會被初始化,并且立刻被初始化為二進制的0,這個動作發(fā)生在所有事件之前,也就是編譯器會立刻將分配給對象的空間初始化。一會的小例子將證明這一點。
***就是調(diào)用類的構(gòu)造方法了。
下面有一個不錯的例子,為了演示成員變量最早被初始化為0了,我們將在父類的構(gòu)造函數(shù)中調(diào)用子類的方法(利用了多態(tài))。
- package fruit;
- import vege.Inner;
- /**
- * @author Octobershiner
- */
- public class Fruit {
- //static block
- static {
- System.out.println("In Fruit static");
- }
- private Inner i = new Inner(); //a private member
- public Fruit(){
- System.out.println("Before Fruit Constructor");
- show(); //由于多態(tài)的特性,此處子類Apple覆寫的方法會被調(diào)用
- System.out.println("After Fruit Constructor");
- }
- public void show(){
- System.out.println("show:Fruit.");
- }
- public static void main(String[] args) {
- // TODO code application logic here
- new Apple(3);
- }
- }
現(xiàn)在父類中須要初始化的有
- 靜態(tài)塊
- 一個Inner類私有成員
- 構(gòu)造函數(shù)
現(xiàn)在我們看子類的代碼
- package fruit;
- public class Apple extends Fruit{
- //靜態(tài)塊
- static{
- System.out.println("In Apple static");
- }
- private int weight = 1; //初始化為1 注意區(qū)別這里和 初始化為0
- public Apple(int para_weight){
- System.out.println("Before Apple Constructer: weight = "+weight);
- weight = para_weight;
- System.out.println("Apple Constructor: weight="+weight);
- }
- @Override
- public void show(){
- System.out.println("show apple: weight =" + weight);
- }
- }
- 靜態(tài)塊
- 私有成員weight
- 構(gòu)造函數(shù)
那么當(dāng)我們運行的時候會有怎樣的結(jié)果呢?猜想。。。。。
下面就是執(zhí)行的結(jié)果:
Look! 首先執(zhí)行父類的靜態(tài)塊,之后是子類的靜態(tài)塊,這兩個應(yīng)該沒有什么問題。接下來就是對父類成員變量的初始化了。首先是父類的私有成員Inner對象,打印了一條“ Inner Constructor”。
接下來就是父類的構(gòu)造函數(shù),可見由于java的多態(tài)性,F(xiàn)ruit的構(gòu)造方法調(diào)用了其子類Apple的show方法,并且我們可以清晰的看到,此刻Apple類中weight變量的值是0!說明,類的成員變量無論是否賦值,在各種初始化之前早已被設(shè)置為二進制0了。
于是乎我想起了很多關(guān)于java的書都在說。。“如果類的私有變量沒有賦值,就會被設(shè)置為0”。。這句話顯然把時間弄混了。。。應(yīng)該是編譯器早已初始化了私有變量,均為0,之后才會執(zhí)行到賦值語句。
父類的構(gòu)造函數(shù)結(jié)束之后,再次回到子類,初始化私有變量(也就是我們常說的賦值語句,因為初始為0的工作早做完了)。所以我們才會看到“Before Apple Constructor weight = 1”,執(zhí)行完構(gòu)造函數(shù)后,我們就看到了weight終于變成了我們創(chuàng)建對象是傳進的3了,呼,初始化結(jié)束。
總 結(jié)
那么總結(jié)一下就是這樣的:
- 編譯器初始化所有的已分配的空間為二進制0 (這是我們的私有變量都會為0,剛才的例子)
- 執(zhí)行父類靜態(tài)代碼 執(zhí)行子類靜態(tài)代碼
- 初始化父類成員變量(我們常說的賦值語句)
- 初始化父類構(gòu)造函數(shù)
- 初始化子類成員變量
- 初始化子類構(gòu)造函數(shù)
原文鏈接:http://www.cnblogs.com/octobershiner/archive/2012/03/12/2391899.html
【編輯推薦】
- Java程序員應(yīng)該了解的10個面向?qū)ο笤O(shè)計原則
- Java集合框架的知識總結(jié)
- Java與F#的并行程序處理對比
- Java并發(fā)編程之同步互斥問題
- Java中String.format的用法