解決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。如有錯誤或未考慮完全的地方,望不吝賜教。

推薦閱讀: