1. 程式人生 > 實用技巧 >java構造器詳解

java構造器詳解

  • 什麼是構造器
    構造器通常也叫構造方法、建構函式,構造器在每個專案中幾乎無處不在。當你new一個物件時,就會呼叫構造器。構造器格式如下:
[修飾符,比如public] 類名 (引數列表,可以沒有引數){
	//這裡不能有return
}
  • 預設構造器
    如果沒有定義構造器,則會預設一個無參構造器,這就是為什麼你定義了一個物件,比如 People,沒有定義任何構造器卻可以new這個物件,比如 new People() 。如果自定義了構造器,則會覆蓋預設構造器。

  • 如何禁止物件被外部建立
    一些特殊需求,不希望定義的物件被外部建立(典型的就是單例了),那直接將構造器的修飾符改為 private 即可。這樣就不能在外部通過new來建立這個物件了。

  • 構造器過載
    與普通方法一樣,構造器也支援過載。一個物件中是可以支援同時定義多個構造器,通過不同的引數列表來實現過載。經常看到程式碼中new一個物件時,有時傳入引數,有時又可以不用傳。比如:new People()跟new People("張三"),這裡就是過載了。

  • 構造器的繼承
    子類構造器會預設呼叫父類無參構造器,如果父類沒有無參構造器,則必須在子類構造器的第一行通過 super關鍵字指定呼叫父類的哪個構造器,具體看下文例子。final類是不允許被繼承的,編譯器會報錯。很好理解,由於final修飾符指的是不允許被修改,而繼承中,子類是可以修改父類的,這裡就產生衝突了,所以final類是不允許被繼承的。

  • 構造器、靜態程式碼塊、構造程式碼塊的執行順序,詳見下文例項

    • 無繼承的情況下的執行順序
      靜態程式碼塊:只在程式啟動後執行一次,優先順序最高
      構造程式碼塊:任何一個構造器被呼叫的時候,都會先執行構造程式碼塊,優先順序低於靜態程式碼塊
      構造器:優先順序低於構造程式碼塊
      總結一下優先順序:靜態程式碼塊 > 構造程式碼塊 > 構造器
    • 有繼承的情況下的執行順序:
      父類靜態程式碼塊:只在程式啟動後執行一次,優先順序最高
      子類靜態程式碼塊:只在程式啟動後執行一次,優先順序低於父類靜態程式碼塊
      父類構造程式碼塊:父類任何一個構造器被呼叫的時候,都會執行一次,優先順序低於子類靜態程式碼塊
      父類構造器:優先順序低於父類構造程式碼
      子類構造程式碼塊
      :子類任何一個構造器被呼叫的時候,都會執行一次,優先順序低於父類構造器
      子類構造器:優先順序低於子類構造程式碼塊
      總結一下優先順序:父類靜態程式碼塊 > 子類靜態程式碼塊 > 父類構造程式碼塊 > 父類構造器 > 子類構造程式碼塊 > 子類構造器

例項

1.預設構造器

新建一個類,不提供任何構造器,編譯器會預設提供一個無參構造器,這就是為什麼沒定義任何構造器,卻可以new 某個物件()

public class People {
    
}

以上這個People類,可以直接通過 new People()來例項化。

2. 禁止物件被外部建立

如果不希望People在外部通過new People()來例項化,只需要將構造器定義為private

public class People {
    private People(){

	}
}

3.構造器過載

過載可以簡單理解為:同個方法名,不同的引數列表。如果希望People能在外部通過new People() 或 new People("字串") 來例項化,則通過以下程式碼即可

public class People {
    //通過new People()呼叫
    public People(){

	}
	//通過new People("字串") 呼叫
	public People(String str){

	}
}

4.構造器的繼承

定義父類構造器,由於該構造器自定義了一個帶參構造器,覆蓋了預設的無參構造器,所以不能直接 new SuperClass() 呼叫了,除非再定義一個無參構造器

/**
 * 父類構造器
 */
public class SuperClass {
    /**
	 * 自定義帶參構造器
	 */
    public SuperClass(String str){
        System.out.println("父類的帶參構造方法,引數為:" + str);
    }
}

定義子類構造器,繼承SuperClass,由於SuperClass沒有無參構造器,所以必須在子類構造器中通過 super("字串")來呼叫,否則編譯器會報錯

/**
 * 子類構造器
 */
public class SubClass extends SuperClass {
    /**
     * 無參構造器
     */
    public SubClass(){
        //由於SuperClass沒有無參構造器,所以必須在子類構造器中通過 super("字串")來呼叫,否則編譯器會報錯。
        //如果沒定義該句,則編譯器會預設呼叫 super()
		super("");
    }
	/**
     * 帶參構造器
     */
    public SubClass(String subStr){
        //由於SuperClass沒有無參構造器,所以必須在子類構造器中通過 super("字串")來呼叫,否則編譯器會報錯。
        //如果沒定義該句,則編譯器會預設呼叫 super()
		super(subStr);
    }
}

5. 構造器、靜態程式碼塊、構造程式碼塊的執行順序

5.1 無繼承的情況

public class People {
    
    static {
        System.out.println("靜態程式碼塊,程式啟動後執行,只會執行一次");
    }

    /**
     * 預設的無參構造器
     */
    public People(){
        System.out.println("預設構造器");
    }

    /**
     * 構造器過載,自定義一個帶參構造器
     * @param str
     */
    public People(String str){
        System.out.println("帶參構造器,引數:" + str);
    }

    {
        System.out.println("構造程式碼塊,每次呼叫構造方法都會執行一次");
    }
}

例項化People

public static void main(String[] args){
    System.out.println("--------------people----------------");
    People people = new People();
    System.out.println("--------------people1----------------");
    People people1 = new People("張三");
}

執行以上程式碼,輸出:

--------------people----------------
靜態程式碼塊,程式啟動後執行,只會執行一次
構造程式碼塊,每次呼叫構造方法都會執行一次
預設構造器
--------------people1----------------
構造程式碼塊,每次呼叫構造方法都會執行一次
帶參構造器,引數:張三

5.2 有繼承的情況

定義父類SuperClass

/**
 * 父類構造器
 */
public class SuperClass {

    {
        System.out.println("父類構造程式碼塊,每次呼叫構造方法都會執行的");
    }

    /**
     * 父類無參構造方法
     */
    public SuperClass(){
        System.out.println("父類的預設構造方法");
    }

    /**
     * 過載,自定義父類帶參構造方法
     * @param str
     */
    public SuperClass(String str){
        System.out.println("父類的帶參構造方法,引數為:" + str);
    }

    static {
        System.out.println("父類的靜態程式碼塊,程式啟動後執行,只會執行一次");
    }

}

定義子類SubClass,繼承SuperClass

/**
 * 子類構造器,繼承SuperClass
 */
public class SubClass extends SuperClass {

    static {
        System.out.println("子類的靜態程式碼塊,程式啟動後執行,只會執行一次,先執行父類的,再執行子類的");
    }

    {
        System.out.println("子類構造程式碼塊,每次呼叫構造方法都會執行的");
    }

    /**
     * 無參構造器
     */
    public SubClass(){

        //這裡沒有指定呼叫父類哪個構造器,會預設呼叫 super(),呼叫父類的無參構造器public SuperClass()
    }

    /**
     * 過載構造器,多傳兩個引數
     * @param str
     * @param str1
     */
    public SubClass(String str,String str1){
        //必須寫在構造器第一行,呼叫父類構造器 public SuperClass(String str)
        super(str);
        System.out.println("子類帶參構造器:" + str1);
    }
}

例項化SubClass

public static void main(String[] args){
    System.out.println("--------------subClass1----------------");
    SubClass subClass1 = new SubClass();

    System.out.println("--------------subClass2----------------");
    SubClass subClass2 = new SubClass("子類第一個引數","子類第二個引數");

}

執行以上程式碼,輸出:

--------------subClass1----------------
父類的靜態程式碼塊,程式啟動後執行,只會執行一次
子類的靜態程式碼塊,程式啟動後執行,只會執行一次,先執行父類的,再執行子類的
父類構造程式碼塊,每次呼叫構造方法都會執行的
父類的預設構造方法
子類構造程式碼塊,每次呼叫構造方法都會執行的
--------------subClass2----------------
父類構造程式碼塊,每次呼叫構造方法都會執行的
父類的帶參構造方法,引數為:子類第一個引數
子類構造程式碼塊,每次呼叫構造方法都會執行的
子類帶參構造器:子類第二個引數