精華區beta b98902HW 關於我們 聯絡資訊
經過一番折騰我終於發現上課時連super都還沒教到,看來得整個從頭講起了...... (不想看我的廢文的話請自行翻閱課本Ch.7) 還記得 this 嗎?一個「指向自身」的reference,例如: class Piano{ public int price; public Piano(int price){ this.price = price; } public void play(String sound){ System.out.println(sound); } public void play(){ this.play("zzz"); } } 可以藉其取用同層 Class 的 fields 及 methods 。 而 super 指向的是「自身 class 所繼承的 class 」,即 superclass 。 但它還是不能存取 superclass 內以 private 宣告的成員。 那它到底有什麼用處呢? 當 superclass 的成員被繼承下來後,subclass 內可能用一個一模一樣的東西, 將其覆寫(override)(用於methods)/隱藏(hidden)(用於fields),而有了新的定義, (如這次作業的 showShort() 及 showAll() ) 但是當我真的非常需要使用上層的東西時呢?就是使用super的時機了。 例如: /** * A conscienceless seller want to disguise a piano as an electric piano. */ class ElectricPiano extends Piano{ public int price; //隱藏了上層的price public ElectricPiano(int price){ super(price); this.price = super.price * 10. } public void play(String sound){ //覆寫了上層的play(String sound) System.out.print("This is the sound played by eletric piano:"); super(sound); } } 新類別的 constructor 中,呼叫了 super(int) ,也就是 superclass 的建構式。 此時在一個 ElectricPiano 的 instance 中,兩個 price 是並存的, 但外界只看得到 subclass 的變數,只有黑心商人能透過 super 取得舊價格, 並將新價格定為它的十倍。 play(String) 被覆寫為新的電鋼琴版本,一樣透過 super 呼叫舊有的 method 。 而 play(無參數) 則未被覆寫,如果客人用到這個函數,黑心商人的詭計就被拆穿啦。 要注意的是, super 不能在 static 函式中使用, 原因就和 static 函式(跟著class)不能取用其他 non-static 成員(跟著object)一樣。 --- 回到原本問題,我明明已經繼承了POOArticle,想來測試一下,為什麼一直拿到hahaha? package ntu.csie.oop10spring; public class POOAdvArticle extends POOArticle{ public static void main(String[] args){ System.out.println("testXD."); } } 表面上看起來沒什麼問題,我什麼事都沒做,為什麼它會說找不到POOArticle() Q_Q。 java 在建構一個 object時,執行的順序是這樣的: 1. 呼叫 superclass 的建構式,將繼承而來的部分建構完成。 2. 將需要初始化的變數初始化。 3. 執行建構式的主體。 很久以前上課有提到,如果沒有定義 constructor ,javac會自動塞一個「空的」進去, 如果有,就使用自行定義的而不會產生空的。 也就是說,如果什麼建構式都沒寫,可以想像成會自動塞進一個像這樣的東西: public POOAdvArticle(){ super(); } 發現問題所在了嗎? 正因為什麼事都沒做,javac編譯時便預設為呼叫無參數的 super(), 但是無參數的 POOArticle() 根本不存在,javac 就這樣自作自受然後對你hahaha。 正規的寫法是在 subclass 建構式第一行就呼叫欲使用的 super() 建構式,像這樣: public class POOAdvArticle extends POOArticle{ ...... public POOAdvArticle(String a, String b, String[] c) { super(a, b, c); ...... } ...... } 如果想要搞怪一點,還可以這樣: public class POOAdvArticle extends POOArticle{ public POOAdvArticle(String[] c) { this("XDD", c); } public POOAdvArticle(String b, String[] c) { this("XD", b, c); } public POOAdvArticle(String a, String b, String[] c) { super(a, b, c); } } 如果在建構式的第一行就遇到呼叫其它建構式,會延緩呼叫 super()的時間。 --- 打這篇花了好久,還是未開始coding T__T, 以上僅提供給大家,有錯或看不懂請鞭,願黑心鋼琴商與你同在。 -- ※ 發信站: 批踢踢實業坊(ptt.cc) ◆ From: 123.204.28.231