首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > JAVA > J2SE开发 >

初始化与内部类的有关问题

2012-01-08 
初始化与内部类的问题看了3天内部类,感觉我开始犯糊涂了,看下下面代码,本来这个code是目的是演示覆盖内部

初始化与内部类的问题
看了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个阶段执行的. 如果没有如上的写法, 你不会感到任何的问题.

热点排行