/**
* 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 的变量时将是父类的变量,而调用方法时将是子类的方法.