1. 程式人生 > 程式設計 >解決java中的父類私有成員變數的繼承問題

解決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.

解決java中的父類私有成員變數的繼承問題

我們看看錯誤的原因。

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子類訪問父類的私有成員變數

子類會繼承父類所有的屬性和方法。

但是根據不同的許可權識別符號,子類不可見父類的私有變數,但可以通過父類的公共方法訪問私有變數

所以對於重名變數,子類和父類都各有一份。

對於子類和父類中重名的方法,則為重寫。即子類重寫了父類的方法,用於多型。

同一個類中函式的簽名不同,則為方法的過載。函式的簽名為函式名+引數列表,與返回值無關。

以上為個人經驗,希望能給大家一個參考,也希望大家多多支援我們。如有錯誤或未考慮完全的地方,望不吝賜教。