Java中的一个坑,在构造方法中调用重写的方法。
首先,我们先来看一段代码:
- 我们创建一个基类和一个派生类,基类的构造方法中调用重写方法。
- 然后我们创建一个派生类对象看一下运行结果:
在运行之前,我们可以来猜测一下结果;我们知道创建一个派生类对象,会先取调基类的构造方法,然后再去调派生类的构造方法。所以这里的运行结果,感觉应该是Base.func()。
可以看到,这里结果是Derive.func(),原因如下:构造D对象的同时,会调用B的构造方法,B的构造方法中调用了func方法,此时会触发动态绑定,会调用到D中的func。 - 我们再将代码略作修改,在派生类中添加一个字段,打印一下字段的值。
可以看到num的值为0,原因和前面一样,在构造d对象的过程中,会先去调用基类的构造方法,此时基类的构造方法中会去调用func方法,触发动态绑定,所以调用的是派生类中重写了的func方法,而此时基类的构造方法还没有调用结束,而private int num = 10;这个代码对num的初始化其实是放在派生类构造方法中完成的,所以在基类构造方法调用完成之前,num是不会被初始化的,所以此时打印出来的num就是没有初始化的默认值0。
结论:
- “用尽量简单的方式使对象进入可工作状态”,尽量不要在构造方法中调用方法(如果这个方法被子类重写,就会触发动态绑定,但是此时子类对象还没有构造完成),可能会出现一些隐藏的但是又极难发现的问题。