解決java中的父類私有成員變量的繼承問題
如果父類中屬性為私有(private),那麼能否被子類繼承呢?
答案是不可以。
我們看如下簡單代碼
class Father { private String name; public void sayHi() { System.out.println("My name is " + this.name); } } class Son extends Father {} public class PrivateFieldTest { public static void main(String[] args) { Father f1 = new Father(); Son s1 = new Son(); f1.sayHi(); s1.sayHi(); } }
得到的結果是:
My name is null My name is null
這裡我們使用的都是默認構造函數,子類自動引用父類的默認構造函數。直接構造為null。
這樣,Son類繼承瞭Father類的sayHi方法,那麼自然,Son的sayHi方法中使用的“name”變量,自然就是Son內部繼承自Father的私有變量name瞭,也就是說私有變量可以被繼承?
不是的。
我們再看下一段代碼,我們在Son的類中重寫一下sayHi方法。
class Father { private String name; public void sayHi() { System.out.println("My name is " + this.name); } } class Son extends Father { public void sayHi() { System.out.println("My name is " + this.name); } } public class PrivateFieldTest { public static void main(String[] args) { Father f1 = new Father(); Son s1 = new Son(); f1.sayHi(); s1.sayHi(); } }
這裡,出現瞭編譯錯誤,即Son類裡面並沒有繼承name.
我們看看錯誤的原因。
The field Father.name is not visible
可以看見,編譯器自動認為,name是屬於Father的,Son內並沒有繼承。
那為什麼第一段代碼中,可以使用sayHi方法得到數據呢?
實際上,這樣解釋比較好:
“子類不能繼承父類的私有屬性,但如果子類中公有的方法影響到瞭父類的私有屬性,那麼私有屬性是能夠被子類使用的。”
這句話聽起來很拗口,但是實際情況確實也很拗口。
看如下代碼
class Father { private String name; public void setName(String name) { this.name = name; } public void sayHi() { System.out.println("My name is " + name); } } class Son extends Father {} public class PrivateFieldTest { public static void main(String[] args) { Father f1 = new Father(); Son s1 = new Son(); f1.sayHi(); s1.sayHi(); System.out.println(); f1.setName("Sam"); f1.sayHi(); s1.sayHi(); System.out.println(); s1.setName("Tom"); f1.sayHi(); s1.sayHi(); } }
運行結果是
My name is null My name is null My name is Sam My name is null My name is Sam My name is Tom
第一段結果,沒有變化。
第二段結果,對應的是我們使用setName方法改變瞭f1的name,所以f1對應的sayHi結果變成瞭Sam。
第三段結果,對應的是我們使用setName方法改變瞭s1的name,所以s1對應的sayHi結果變成瞭Tom。
由此我們可以看到,雖然子類不能繼承父類私有變量,但是還是可以通過公有方法使用私有變量。隻是重寫函數的時候可能比較麻煩,所以要活用super。
補充:Java子類訪問父類的私有成員變量
子類會繼承父類所有的屬性和方法。
但是根據不同的權限標識符,子類不可見父類的私有變量,但可以通過父類的公共方法訪問私有變量
所以對於重名變量,子類和父類都各有一份。
對於子類和父類中重名的方法,則為重寫。即子類重寫瞭父類的方法,用於多態。
同一個類中函數的簽名不同,則為方法的重載。函數的簽名為函數名+參數列表,與返回值無關。
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。如有錯誤或未考慮完全的地方,望不吝賜教。