神秘的类 一件私事
一、概述
在线的程序当中,子类的存在和父类同名成员变量的时候,会打印什么内容呢?
public class PriveMatter {
public static void main(String[] args) {
System.out.println(new Derived().className);
}
}
class Base{
public String className = "Base";
}
class Derived extends Base{
private String className = "Derived";
}
对于程序的表名而言,他会打印出 Derived 因为这个正是存在于每个 Derived 对象实例的 className 当中的内容。
更加升入一点分析的时候,会认为 Derived 不能够编译通过,因为 Derived 中的 className 变量具有比 Base 当中的 className 变量更加具有限定性访问的权限。
如果尝试着编译该程序,就会发现这种分析也不是正确的,该程序不能够编译通过,但是错误却出现在 PrivateMatter当中。
在我们的程序中便于错误出现在 PrivateMatter 类试图访问 Drived.className 的时候,尽管 Base 有一个公共域 className 引用的是私有域 Drived.className 因为这个域声明是 private的,所以它对于 PrivateMatter 来说是不可以访问的,因此 编译器会出现一条错误的信息:
PrivateMatter.java 11:className has private access in Derived
System.out.println(new Derived().className);
二、注意
尽管我们的 Derived 实例当中的公共域对象 Base.className 被隐藏了,但是我们可以通过 Derived 实例转型成为 Base 来访问到它,下面的版本当中就可以打印出 Base
public static void main(String[] args) {
System.out.println(((Base)new Derived()).className);
}
这个说明覆盖与隐藏之间存在区别,一旦一个方法在子类当中覆盖重写,你就不能够使用子类的对象向上转型调用到父亲的方法了。(除非你在子类重写方法当中,采用super关键字调用)
然而,你可以通过将子类实例转换成为某个父类的类型来访问隐藏的域,在这个父类中该域未被覆盖。
三、解决
如果你想让这个程序打印 Derived 也就是说,想要展示重写行为,那么你可以用公共方法来替代公共域。在任何情况下,这都是一个好主意。
public class PriveMatter {
public static void main(String[] args) {
System.out.println(new Derived().getClassName());
}
}
class Base{
public String getClassName(){
return "Base";
}
}
class Derived extends Base{
@Override
public String getClassName(){
return "Base";
}
}
对于语言设计者而言,应该考虑到消除隐藏性的可能性:
例如:使用所有的域都应该是隐含私有的,如果这样做显得过于严苛,那么至少应该考虑到对隐藏进行限制,以使其遵循包容原则。
四、总结
当我们声明一个域、一个静态方法或者嵌套类的时候,如果其名称和父类中对应的可以访问得到的域、方法、或者类型相同的情况,就会发生隐藏。
隐藏是容易产生混乱的,违背包容性的隐藏域在某种意义上是特别有害的。一般来讲,除了覆盖重写之外,尽量避免名字重用。