1. 程式人生 > >Java關鍵字(六)——super

Java關鍵字(六)——super

限定 父類 除了 bsp inf clas 了解 顯示 TE

  在 Java關鍵字(五)——this 中我們說 this 關鍵字是表示當前對象的引用。而 Java 中的 super 關鍵字則是表示 父類對象的引用。

  我們分析這句話“父類對象的引用”,那說明我們使用的時候只能在子類中使用,既然是對象的引用,那麽我們也可以用來調用成員屬性以及成員方法,當然了,這裏的 super 關鍵字還能夠調用父類的構造方法。具體有如下幾種用法:

1、調用父類的構造方法

  Java中的繼承大家都應該了解,子類繼承父類,我們是能夠用子類的對象調用父類的屬性和方法的,我們知道屬性和方法只能夠通過對象調用,那麽我們可以大膽假設一下:

  在創建子類對象的同時,也創建了父類的對象,而創建對象是通過調用構造函數實現的,那麽我們在創建子類對象的時候,應該會調用父類的構造方法。

  下面我們看這段代碼:

 1 public class Parent {
 2 
 3     public Parent(){
 4         System.out.println("父類默認無參構造方法");
 5     }
 6 }
 7 
 8 
 9 public class Son extends Parent {
10 
11     public Son(){
12         System.out.println("子類默認無參構造方法");
13 } 14 }

  下面我們創建子類的對象:

1     public static void main(String[] args) {
2         Son son = new Son();
3     }

  打印結果:

  技術分享圖片

  通過打印結果看到我們在創建子類對象的時候,首先調用了父類的構造方法,接著調用子類的構造方法,也就是說在創建子類對象的時候,首先創建了父類對象,與前面我們猜想的一致。

  那麽問題又來了:是在什麽時候調用的父類構造方法呢?

  可以參考Java官方文檔:https://docs.oracle.com/javase/specs/jls/se8/html/jls-8.html#d5e14278

  技術分享圖片

  

  紅色框內的英文翻譯為:如果聲明的類是原始類Object,那麽默認的構造函數有一個空的主體。否則,默認構造函數只是簡單地調用沒有參數的超類構造函數。

  也就是說除了頂級類 Object.class 構造函數沒有調用父類的構造方法,其余的所有類都默認在構造函數中調用了父類的構造函數(沒有顯式聲明父類的子類其父類是 Object)。

  那麽是通過什麽來調用的呢?我們接著看官方文檔:

技術分享圖片

  上面的意思大概就是超類構造函數通過 super 關鍵字調用,並且是以 super 關鍵字開頭。

  所以上面的 Son 類的構造方法實際上應該是這樣的:

  技術分享圖片

  ①、子類默認是通過 super() 調用父類的無參構造方法,如果父類顯示聲明了一個有參構造方法,而沒有聲明無參構造方法,實例化子類是會報錯的。

 1 public class Parent {
 2 
 3     public Parent(String name){
 4         System.out.println("父類有參構造方法");
 5     }
 6 }
 7 
 8 public class Son extends Parent {
 9 
10     public Son(){
11         System.out.println("子類默認無參構造方法");
12     }
13 
14     public static void main(String[] args) {
15         Son son = new Son();
16     }
17 
18 }

  上面代碼是會報錯的:

  技術分享圖片

  解決辦法就是通過 super 關鍵字調用父類的有參構造方法:

 1 public class Son extends Parent {
 2 
 3     public Son(){
 4         super("Tom");
 5         System.out.println("子類默認無參構造方法");
 6     }
 7 
 8     public static void main(String[] args) {
 9         Son son = new Son();
10     }
11 
12 }

  註意看第 4 行代碼,同理,多個參數也是這種調法。

2、調用父類的成員屬性

 1 public class Parent {
 2     public String name;
 3 
 4     public Parent(){
 5         System.out.println("父類默認無參構造方法");
 6     }
 7 }
 8 
 9 public class Son extends Parent {
10 
11     public Son(){
12         System.out.println("子類默認無參構造方法");
13     }
14 
15     public void printName(){
16         System.out.println(super.name);
17     }
18 
19 }

  第 16 行代碼 super.父類屬性 通過這種形式來調用父類的屬性。

3、調用父類的方法

 1 public class Parent {
 2     public String name;
 3 
 4     public Parent(){
 5         System.out.println("父類默認無參構造方法");
 6     }
 7 
 8     public void setName(String name){
 9         this.name = name;
10     }
11 }
12 
13 public class Son extends Parent {
14 
15     public Son(){
16         super();//1、調用父類構造函數
17         System.out.println("子類默認無參構造方法");
18     }
19 
20     public void printName(){
21         super.setName("Tom");//2、調用父類方法
22         System.out.println(super.name);//3、調用父類屬性
23     }
24 
25     public static void main(String[] args) {
26         Son son = new Son();
27         son.printName();//Tom
28     }
29 
30 }

  這個例子我們在子類中分別調用了父類的構造方法、普通方法以及成員屬性。

4、this 和 super 出現在同一個構造方法中?

  不能!!!

  在上一篇博客對 this 關鍵字 的介紹中,我們知道能夠通過 this 關鍵字調用自己的構造方法。而本篇博客介紹 super 關鍵字,我們知道了能夠通過 super 調用父類的構造方法,那麽這兩個關鍵字能同時出現在子類的構造方法中嗎?

  ①、假設 super() 在 this() 關鍵字的前面

  首先通過 super() 調用父類構造方法,對父類進行一次實例化。接著調用 this() ,this() 方法會調用子類的構造方法,在子類的構造方法中又會對父類進行一次實例化。也就是說我們對子類進行一次實例化,對造成對父類進行兩次實例化,所以顯然編譯器是不允許的。

 1 public class Parent {
 2     public String name;
 3 
 4     public Parent(){
 5         System.out.println("父類默認無參構造方法");
 6     }
 7 
 8     public Parent(String name){
 9         System.out.println("父類有參構造方法");
10     }
11 
12 }
13 
14 public class Son extends Parent {
15 
16     public Son(){
17         super();//1、調用父類構造函數
18         this("Tom");//2、調用子類構造方法
19         System.out.println("子類默認無參構造方法");
20     }
21 
22     public Son(String name){
23         System.out.println("子類有參構造方法");
24     }
25 
26 }

  反過來 this() 在 super() 之前也是一樣。

  而且編譯器有限定 this() 和 super() 這兩個關鍵字都只能出現在構造方法的第一行,將這兩個關鍵字放在一起,總有一個關鍵字在第二行,編譯是不能通過的。   

Java關鍵字(六)——super