解決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 nullMy 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 nullMy name is nullMy name is SamMy name is nullMy name is SamMy name is Tom
第一段結果,沒有變化。
第二段結果,對應的是我們使用setName方法改變了f1的name,所以f1對應的sayHi結果變成了Sam。
第三段結果,對應的是我們使用setName方法改變了s1的name,所以s1對應的sayHi結果變成了Tom。
由此我們可以看到,雖然子類不能繼承父類私有變量,但是還是可以通過公有方法使用私有變量。只是重寫函數的時候可能比較麻煩,所以要活用super。
補充:Java子類訪問父類的私有成員變量
子類會繼承父類所有的屬性和方法。
但是根據不同的權限標識符,子類不可見父類的私有變量,但可以通過父類的公共方法訪問私有變量
所以對于重名變量,子類和父類都各有一份。
對于子類和父類中重名的方法,則為重寫。即子類重寫了父類的方法,用于多態。
同一個類中函數的簽名不同,則為方法的重載。函數的簽名為函數名+參數列表,與返回值無關。
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持好吧啦網。如有錯誤或未考慮完全的地方,望不吝賜教。
相關文章:
1. Java發送http請求的示例(get與post方法請求)2. JS繪圖Flot如何實現動態可刷新曲線圖3. springboot基于Redis發布訂閱集群下WebSocket的解決方案4. 關于探究python中sys.argv時遇到的問題詳解5. 基于android studio的layout的xml文件的創建方式6. CSS自定義滾動條樣式案例詳解7. 使用ProcessBuilder調用外部命令,并返回大量結果8. Intellij IDEA官方最完美編程字體Mono使用9. python使用requests庫爬取拉勾網招聘信息的實現10. IDEA項目的依賴(pom.xml文件)導入問題及解決
