父类引用指向子类变量

 
public class X
{
int i =1;
void method()
{
System.out.println("1");
}
}
class Y extends X
{
int i = 2;
void method()
{
System.out.println("2");
}
public static void main(String[] args)
{
X s = new Y();
System.out.println(s.i);
s.method();
}
}


我想之所以对java的这种行为感到奇怪,是因为你把类里的属性和方法来做了对比吧。也就是子类(Y)同时覆盖了父类(X)的属性和方法时,为什么“X s = new Y(); ”这里的s只可以调用有Y中覆盖X的方法,而不能使用Y中覆盖X的属性。这也是疑惑的关键。 我先给出一个结论:父类引用指向子类对象时,父类与子类同名的属性和方法都会被覆盖。 (一般对属性的覆盖叫对属性的隐藏) 对于上面的结论我们来看你这个程序,既然属性可以被覆盖,那怎么s.i是用了父类的变量值呢?这是关系到了java处理属性和方法的不同机制引起的。 先再给出一个结论:java对方法的调用是采取动态绑定(运行时才判断)的方法,对属性的调用是采取静态绑定(编译时就判断)的方法。 对于上面的结论我们来理解你的程序“X s = new Y(); ”,这一句,也可说是父类引用指向子类对象,也可以说是子类对象向上转型为父类引用。那在s使用属性的时候是静态绑定的,它是在编译期就决定了这个属性是属于哪一个类的对象了,所以在上面那句中X s.也就是说X类中的属性是用s来调用的,s就不能调用Y类中的属性,因为它是静态绑定了。而s使用方法时是动态绑定的,所以s在调用方法的时候,java会去查看s这个时候的指向的对象是哪一个,是Y类对象就调用Y类的方法,是X类的对象就调用X类的方法。所以这里s是指向了Y类中的对象(“X s = new Y(); ”这句可知),则调用Y类中的方法。 好,上面对你的问题讲清了,下面再说说java因为什么对属性要采取静态的绑定方法。这是因为静态绑定是有很多的好处,它可以让我们在编译期就发现程序中的错误,而不是在运行期。这样就可以提高程序的运行效率!而对方法采取动态绑定是为了实现多态,知道java是的多态是java的一大特色。多态也是面向对象的关键技术之一,所以java是以效率为代价来实现多态这是很值得的
相关文章
相关标签/搜索