这里只说一个完整的结果至于为什么是这样的顺序可以参考我以前的文章深入剖析java类的构造方式
如果父类有静态成员赋值或者静态初始化块执行静态成员赋值和静态初始化块
如果类有静态成员赋值或者静态初始化块执行静态成员赋值和静态初始化块
将类的成员赋予初值(原始类型的成员的值为规定值例如int型为float型为fboolean型为false对象类型的初始值为null)
如果构造方法中存在this()调用(可以是其它带参数的this()调用)则执行之执行完毕后进入第步继续执行如果没有this调用则进行下一步(这个有可能存在递归调用其它的构造方法)
执行显式的super()调用(可以是其它带参数的super()调用)或者隐式的super()调用(缺省构造方法)此步骤又进入一个父类的构造过程并一直上推至Object对象的构造
执行类申明中的成员赋值和初始化块
执行构造方法中的其它语句
其中第步是比较麻烦的因为this调用实际上会调用类的另外一个构造方法最终应该是执行类的某个构造方法它可能会显示的调用super但是无论是否调用super最终都是执行super的也就是父类的构造方法并一直这样递归到Object所以在子类和父类的构造中首先构造或者说执行的是父类的构造但是它是由子类的构造方法调用的先于构造方法的方法体里面的内容这个是由编译器决定的所以我感觉简单直观一些的顺序表述应该是
如果父类有静态成员赋值或者静态初始化块执行静态成员赋值和静态初始化块
如果类有静态成员赋值或者静态初始化块执行静态成员赋值和静态初始化块
将类的成员赋予初值(原始类型的成员的值为规定值例如int型为float型为fboolean型为false对象类型的初始值为null)
执行构造方法并可能递归调用this()最终先执行父类的构造方法并一直递归到Object的构造方法的执行
父类的构造方法执行完成后执行类申明中的成员赋值和初始化块
执行构造方法中的其它语句
最终的简化顺序版本是
父类的静态成员赋值和静态块
子类的静态成员和静态块
父类的构造方法
父类的成员赋值和初始化块
父类的构造方法中的其它语句
子类的成员赋值和初始化块
子类的构造方法中的其它语句
年月日更新
针对留言中提到的那个文章中的问题发现这个顺序也是有不足的情况这个顺序是一般的顺序但是有可能被打破留言中的那篇文章就是一个例子因为在执行静态初始化块的时候先执行了类的构造打破了这个一般顺序所以这个顺序有个前提就是静态赋值和初始化块中没有对本类的实例化语句
对于那个文章中的问题作者最后的解决方法可行但是不见得是最好的可以简单的修改静态赋值和静态初始化块的顺序修改后的代码片断为
public class CachingEnumResolver {
private static Map CODE_MAP_CACHE;
/*MSGCODE>Category内存索引*/
static {
CODE_MAP_CACHE = new HashMap();
//为了说明问题我在这里初始化一条数据
CODE_MAP_CACHEput(北京市);
}
//单态实例一切问题皆由此行引起
private static final CachingEnumResolver SINGLE_ENUM_RESOLVER = new CachingEnumResolver();