Java基礎之繼承方面的學習(三)
上面的博文已經寫了大部分了,下面這一篇博文我們主要寫繼承之後,父類和子類的方法的覆蓋和隱藏方面。
該文章版署源自:
源自連線:https://www.cnblogs.com/dolphin0520/p/3803432.html
作者:海子
你真的瞭解了繼承的思想了嘛?瞭解了繼承的過程了嘛?
學以致用:
下面這段程式碼的輸出結果是什麼?
public class Test { public static void main(String[] args) { new Circle(); } } class Draw { public Draw(String type) { System.out.println(type+" draw constructor"); } } class Shape { private Draw draw = new Draw("shape"); public Shape(){ System.out.println("shape constructor"); } } class Circle extends Shape { private Draw draw = new Draw("circle"); public Circle() { System.out.println("circle constructor"); } }
shape draw constructor
shape constructor
circle draw constructor
circle constructor
這道題目主要考察的是類繼承時構造器的呼叫順序和初始化順序。要記住一點:父類的構造器呼叫以及初始化過程一定在子類的前面。由於Circle類的父類是Shape類,所以Shape類先進行初始化,然後再執行Shape類的構造器。接著才是對子類Circle進行初始化,最後執行Circle的構造器。 |
2.下面這段程式碼的輸出結果是什麼?
public class Test { public static void main(String[] args) { Shape shape = new Circle(); System.out.println(shape.name); System.out.println(shape.getName()); shape.printType(); shape.printName(); } } class Shape { public String name = "shape"; public void setName(String name){ this.name = name; } public String getName(){ return this.name; } public Shape(){ System.out.println("shape constructor"); } public void printType() { System.out.println("this is shape"); } public static void printName() { System.out.println("shape"); } } class Circle extends Shape { public String name = "circle"; public void setName(String name){ this.name = name; } public String getName(){ return this.name; } public Circle() { System.out.println("circle constructor"); } public void printType() { System.out.println("this is circle"); } public static void printName() { System.out.println("circle"); } }
shape constructor
circle constructor
shape
circle
this is circle
shape
這道題主要考察了隱藏和覆蓋的區別(當然也和多型相關)。
覆蓋只針對非靜態方法(終態方法不能被繼承,所以就存在覆蓋一說了),而隱藏是針對成員變數和靜態方法的。這2者之間的區別是:覆蓋受RTTI(Runtime type identification)約束的,而隱藏卻不受該約束。也就是說只有覆蓋方法才會進行動態繫結,而隱藏是不會發生動態繫結的。在Java中,除了static方法和final方法,其他所有的方法都是動態繫結。因此,就會出現上面的輸出結果。
學習總結:假如A類被B類繼承了,A類指向了建立B類的地址的時候,呼叫以下方面的時候實際呼叫的是B類中的值或者方法。
-----》A類被B類的重寫的方法,在呼叫的時候,實際呼叫的是B類的方法。
(上面解析:Shape類的name,Circle類中也有,但是setName()方法重寫了name的值,所以Shape類輸出name的時候,呼叫的其實是Circle類的值。但是Shape類的name還是shape)