初始化与内部类的问题
看了3天内部类,感觉我开始犯糊涂了,看下下面代码,本来这个code是目的是演示覆盖内部类的,然而这个用法我认为确实很容易理解 ,但是我糊涂了他的整个代码的初始化过程,我可能对继承,内部类概念搞混了看下面代码:
import com.bruceeckel.simpletest.*;
class Egg2 {
protected class Yolk {
public Yolk() { System.out.println( "Egg2.Yolk() "); }
public void f() { System.out.println( "Egg2.Yolk.f() ");}
}
private Yolk y = new Yolk();
public Egg2() { System.out.println( "New Egg2() "); }
public void insertYolk(Yolk yy) { y = yy; }
public void g() { y.f(); }
}
public class BigEgg2 extends Egg2 {
private static Test monitor = new Test();
public class Yolk extends Egg2.Yolk {
public Yolk() { System.out.println( "BigEgg2.Yolk() "); }
public void f() {
System.out.println( "BigEgg2.Yolk.f() ");
}
}
public BigEgg2() { insertYolk(new Yolk()); }
public static void main(String[] args) {
Egg2 e2 = new BigEgg2();
e2.g();
monitor.expect(new String[] {
"Egg2.Yolk() ",
"New Egg2() ",
"Egg2.Yolk() ",
"BigEgg2.Yolk() ",
"BigEgg2.Yolk.f() "
});
}
}
这里首先打印出的是 "Egg2.Yolk() ",我的初始化思路是 首先产生了一个Y对象 private Yolk y = new Yolk();所以他先调用了基类内部类构造器中 public Yolk() { System.out.println( "Egg2.Yolk() "); }第一个 "Egg2.Yolk() ",就输出了,之后
Egg2 e2 = new BigEgg2(),e2产生了BigEgg2类的对象并向上转型为基类Egg2,首先先调用基类构造器, public Egg2() { System.out.println( "New Egg2() "); }第2个结果输出了 "New Egg2() ",然后是导出类的构造器初始化但构造器public BigEgg2() { insertYolk(new Yolk()); }却产生了一个内部类的一个对象,然而这个对象的创建首先调用了其基类Yolk 的构造器 public Yolk() { System.out.println( "Egg2.Yolk() "); }结果输出打印了第3个值 "Egg2.Yolk() ",之后就开始对导出类Yolk的初始化public Yolk() { System.out.println( "BigEgg2.Yolk() "); }输出了第四个BigEgg2.Yolk() ",由于insertYolk()方法 将BigEgg2() 中Yolk对象向上转型为Y,通过后期动态绑定,调用g(),方法中f实际调用的是public void f() {
System.out.println( "BigEgg2.Yolk.f() ");,最后 "BigEgg2.Yolk.f() "的结果输出了,我是这样理解的,不知道思路对不对,我现在对继承与基类,外部类与内部类他们之间关系的差异弄混了,请高手指点下!
[解决办法]
楼主真是太巧了,今天我也刚刚看到这里,也是这本书,我也头疼啊,高手支招把
[解决办法]
这个其实和内部类没什么关系吧?
实例化类的时候先实例化参数(Field),然后构造(constructor)。
构造的时候先构造父类,构造父类前先实例化父类的参数。如此循环。
有一个问题是如下代码
abstract class A {
public A() {
createA();
}
public abstract A createA();
}
class B extends A {
private A a = null;
public A createA() {
this.a = new B();
return this.a;
}
}
在createA方面跟踪的时候你会发现B.a的确!=null, 但是在你使用B的其他方法用到B.a的时候发现是null.
最后B.a是null.
因为初始化B.a, 构造B的时候调用了A的构造, 调用了createA方法,实际上是B里面实现的, 给B.a赋值. 然后构造B, 实例化B.a,也就是执行a = null;这句话, 最后B.a变成null.
从这里可以看到private A a = null;这句话实际上是分2个阶段执行的. 如果没有如上的写法, 你不会感到任何的问题.