问题描述
在Java中,子类能否继承父类的私有成员?
官方解答
我们来看看官方的解答,链接如下:https://docs.oracle.com/javase/specs/jls/se8/html/jls-8.html#jls-8.2
官方文档中明确描述:Members of a class that are declared private
are not inherited by subclasses of that class. 也就是说:子类不可继承父类中被private修饰的成员!
看到这里,或许还有点疑惑,比如接下来这个例子。
疑惑示例
父类如下:
package com.test;
/**
* 本文作者:谷哥的小弟
* 博客地址:http://blog.csdn.net/lfdfhl
*/
public class Father {
// 父类的私有属性
private String name;
// name属性的get方法
public String getName() {
return name;
}
// name属性的set方法
public void setName(String name) {
this.name = name;
}
}
子类如下:
package com.test;
/**
* 本文作者:谷哥的小弟
* 博客地址:http://blog.csdn.net/lfdfhl
*/
public class Son extends Father {
}
测试如下:
package com.test;
/**
* 本文作者:谷哥的小弟
* 博客地址:http://blog.csdn.net/lfdfhl
*/
public class Test {
public static void main(String[] args) throws Exception {
Son son = new Son();
son.setName("张三");
System.out.println(son.getName());
}
}
问题如下:
既然name属性是父类的私有成员,为什么子类对象可以调用setName( )和getName( )方法呢?
而且,请注意父类的setName( )方法:
public void setName(String name) {
this.name = name;
}
在该方法中使用了this.name = name。我们知道:this代表的是当前对象。那么,就是说:子类对象实际上是有name属性的?
那就奇怪了:name属性在父类中是被private修饰的怎么可能被子类继承呢?
答疑解惑
其实:JVM创造子类对象时,首先会在内存中创建一个父类对象。然后,在父类对象外部放上子类独有的成员。这两者合起来形成最终的子类对象。
也就是说:子类继承父类之后;子类拥有了父类的所有成员,包括私有成员。但是,这并不代表着子类可以直接访问这些私有成员。
不过,子类可以通过父类私有属性对应的公共的get和set方法间接访问父类私有属性!
这一点在Java语法层面上也有所体现:子类构造函数的第一行默认调用其父类的构造函数。即,先创建父类对象再创建子类对象。
参考资料
https://stackoverflow.com/questions/4716040/do-subclasses-inherit-private-fields
https://www.dazhuanlan.com/2019/12/14/5df3cf91b95b0/
https://docs.oracle.com/javase/specs/jls/se8/html/jls-8.html#jls-8.2
https://blog.csdn.net/renfufei/article/details/95758333