关闭

访问子类对象的实例变量

275人阅读 评论(0) 收藏 举报
分类:
/**
 * 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初始化为222
	public 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.i
		System.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.i
		System.out.println(this.i);
		this.display();
		// 输出this实际的类型,将看到输出Derived
		System.out.println(this.getClass());
		// 因为this的编译类型是Base,所以依然不能调用sub()方法,
		// this.sub();
	}
上面程序调用this.getClass()来获取this代表的对象,将看到输出Derived类,这代表此时this引用代表的是Derived对象.但是却无法调用sub()方法,编译出错,这就是因为this的编译时类型是Base的缘故.

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


0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:8911次
    • 积分:215
    • 等级:
    • 排名:千里之外
    • 原创:11篇
    • 转载:9篇
    • 译文:0篇
    • 评论:0条
    文章分类