受保护的成员从java中的不同包访问 – 一个好奇心

package packageOne;
public class Base
{
protected void display(){
system.out.println("in Base");
}
}


package packageTwo;
public class Derived extends packageOne.Base{
public void show(){
new Base().display();//this is not working throws compilation error that display() from the type Base is not visible
new Derived().display();//is working
display();//is working
}
}

这两个包在两个不同的文件中.但是为什么会这样呢?

解决方法

http://java.sun.com/docs/books/jls/third_edition/html/names.html#6.6
class C
    protected member;

// in a different package

class S extends C 

    obj.member; // only allowed if type of obj is S or subclass of S

动机可能如下.如果obj是S,S类具有足够的内部知识,它有权操纵其成员,并且可以安全地执行此操作.

如果obj不是S,它可能是C的另一个子类S2,S不知道. S写的时候,S2可能还没有出生.对于S来操纵S2的受保护的内部部件是相当危险的.如果这是允许的,从S2的角度来看,它不知道谁将篡改其受保护的内部部件,如何,这使得S2工作非常难以推理自己的状态.

现在,如果obj是D,D扩展S,S是否有危险访问obj.member?不是真的. S使用成员是S及其所有子类的共享知识,包括D.S作为超类有权定义行为,D作为子类有义务接受和遵守.

为了更容易理解,规则应该被简化,以使obj(静态)类型完全为S.毕竟,子类D出现在S中是非常不寻常和不合适的.即使发生,静态类型的obj是D,我们的简化规则可以通过上传容易地处理:((S)obj).成员

相关文章

Java中的String是不可变对象 在面向对象及函数编程语言中,不...
String, StringBuffer 和 StringBuilder 可变性 String不可变...
序列化:把对象转换为字节序列的过程称为对象的序列化. 反序...
先说结论,是对象!可以继续往下看 数组是不是对象 什么是对...
为什么浮点数 float 或 double 运算的时候会有精度丢失的风险...
面试题引入 这里引申出一个经典问题,看下面代码 Integer a ...