解决java中的父类私有成员变量的继承问题
作者:摘下singsing
如果父类中属性为私有(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子类访问父类的私有成员变量
子类会继承父类所有的属性和方法。
但是根据不同的权限标识符,子类不可见父类的私有变量,但可以通过父类的公共方法访问私有变量
所以对于重名变量,子类和父类都各有一份。
对于子类和父类中重名的方法,则为重写。即子类重写了父类的方法,用于多态。
同一个类中函数的签名不同,则为方法的重载。函数的签名为函数名+参数列表,与返回值无关。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。如有错误或未考虑完全的地方,望不吝赐教。