解決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子類訪問父類的私有成員變數
子類會繼承父類所有的屬性和方法。
但是根據不同的許可權識別符號,子類不可見父類的私有變數,但可以通過父類的公共方法訪問私有變數
所以對於重名變數,子類和父類都各有一份。
對於子類和父類中重名的方法,則為重寫。即子類重寫了父類的方法,用於多型。
同一個類中函式的簽名不同,則為方法的過載。函式的簽名為函式名+引數列表,與返回值無關。
以上為個人經驗,希望能給大家一個參考,也希望大家多多支援我們。如有錯誤或未考慮完全的地方,望不吝賜教。