访问子类对象的实例变量

来源:互联网 发布:ubuntu gcc 版本 编辑:程序博客网 时间:2024/05/29 10:36
/** * Description: * <br/>网站: <a href="http://www.crazyit.org">疯狂Java联盟</a> * <br/>Copyright (C), 2001-2014, Yeeku.H.Lee * <br/>This program is protected by copyright laws. * <br/>Program Name: * <br/>Date: * @author Yeeku.H.Lee kongyeeku@163.com * @version 1.0 */class Base{// 定义了一个名为i的实例变量private int i = 2;public Base(){this.display();}public void display(){System.out.println(i);}}// 继承Base的Derived子类class Derived extends Base{// 定义了一个名为i的实例变量private int i = 22;// 构造器,将实例变量i初始化为222public Derived(){i = 222;              //②}public void display(){System.out.println(i);}        public void sub(){} }public class Test{public static void main(String[] args){// 创建Derived的构造器创建实例new Derived();       //①}}


输出结果:0

在生成子类实例时先调用父类的构造器,生成父类的实例,构造器的调用是完成java对象的初始化,但在此之前java对象已经分配好 内存空间-----0或nul.

在执行Derived的构造器之前系统先调用Base的构造器,此时 一行代码 this.display(); 经过编译处理应该包含两行代码i=2;this.display();但此时this代表什么?

先看如下代码:

public Base(){// 直接输出this.iSystem.out.println(this.i);this.display();}
此时输出2 , 0

为什么?

当this在构造器中时,this代表正在初始化的java对象.但此时,this位于Base构造器内,但这些代码实际放在Derived的构造器中------是Derived的构造器隐式调用Base构造器的代码.

由此可见this代表Derived对象.

但是为什么上例会输出2呢?

这是因为:这个this虽然代表Derived对象,但是却位于Base构造器中,它的编译类型是Base,实际引用的是一个Derived对象(父类引用指向子类对象).为证明这一点,再次改写程序


public Base(){// 直接输出this.iSystem.out.println(this.i);this.display();// 输出this实际的类型,将看到输出DerivedSystem.out.println(this.getClass());// 因为this的编译类型是Base,所以依然不能调用sub()方法,// this.sub();}
上面程序调用this.getClass()来获取this代表的对象,将看到输出Derived类,这代表此时this引用代表的是Derived对象.但是却无法调用sub()方法,编译出错,这就是因为this的编译时类型是Base的缘故.

当变量的编译时类型和运行时类型不一样时,通过访问它引用的实例对象时,该实例对象的值是由声明该变量的类型决定.但通过该变量调用它引用的对象的方法时,将会调用实际引用的对象的方法.即父类引用(Father father)指向子类对象(Sun sun)时,调用father 的变量时将是父类的变量,而调用方法时将是子类的方法.


原创粉丝点击