我们知道一个类要被使用必须经过装载、连接、初始化这样的过程,下面先对这3个过程进行一个简单的概述。
在装载阶段类装载器将编译形成的class文件装载禁内存,创建类相关的class对象,这个class对象封装了我们要使用的类的类型信息。
连接阶段又可以被分为3个阶段,验证、准备、解析。
验证就是确保java类型格式的正确性,并适于jvm使用。
准备阶段,jvm为静态变量分配空间,并设置默认值,这里只设置默认值,例如int型变量会被赋予默认值0。在这一阶段,jvm可能还会为一些数据结构分配内存,目的是提高运行程序的性能,比如说方法表。
解析过程就是在类型的常量池中寻找类、接口、字段和方法的符号引用,把这些符号引用替换成直接引用。这个阶段可以被推迟到初始化之后,在程序运行中真正使用某个符号引用时再去解析它。
类初始化的顺序下面这个程序完整的展现出来了,而且十分易记。
class Test{
static{
System.out.println("父类static 块 1 执行");
}
static Sample staticSam1=new Sample("父类 静态成员staticSam1初始化");
Sample sam1=new Sample("父类 sam1成员初始化");
static Sample staticSam2=new Sample("父类 静态成员staticSam2初始化");
static{
System.out.println("父类 static 块 2 执行");
}
Test()
{
System.out.println("父类 Test默认构造<span id="21_nwp" style="width: auto; height: auto; float: none;"><a id="21_nwl" href="http://cpro.baidu.com/cpro/ui/uijs.php?adclass=0&app_id=0&c=news&cf=1001&ch=0&di=128&fv=11&is_app=0&jk=46fc72f8b03d1143&k=%BA%AF%CA%FD&k0=%BA%AF%CA%FD&kdi0=0&luki=8&mcpm=0&n=10&p=baidu&q=xiaoxiaobai_cpr&rb=0&rs=1&seller_id=1&sid=43113db0f872fc46&ssp2=1&stid=9&t=tpclicked3_hc&td=1972324&tu=u1972324&u=http%3A%2F%2Fwww%2E52ij%2Ecom%2Fjishu%2Fjava%2F98821%2Ehtml&urlid=0" target="_blank" mpid="21" style="text-decoration: none;"><span style="color:#0000ff;font-size:12px;width:auto;height:auto;float:none;">函数</span></a></span>被调用");
}
Sample sam2=new Sample("父类 sam2成员初始化");
}
class TestSub extends Test
{
static Sample staticSamSub=new Sample("子类 静态成员staticSamSub初始化");
TestSub()
{
System.out.println("子类 TestSub 默认构造函数被调用");
}
Sample sam1=new Sample("子类 sam1成员初始化");
static Sample staticSamSub1=new Sample("子类 静态成员staticSamSub1初始化");
static{System.out.println("子类 static 块 执行");}
Sample sam2=new Sample("子类 sam2成员初始化");
}
执行结果:
父类 static 块 1 执行
父类 静态成员staticSam1初始化
父类 静态成员staticSam2初始化
父类 static 块 2 执行
--------父类静态成员初始化
子类 静态成员staticSamSub初始化
子类 静态成员staticSamSub1初始化
子类 static 块 执行
-------子类静态成员初始化
父类 sam1成员初始化
父类 sam2成员初始化
父类 Test默认构造函数被调用
-------父类普通成员初始化和构造函数执行
子类 sam1成员初始化
子类 sam2成员初始化
子类 TestSub 默认构造函数被调用
-------父类普通成员初始化和构造函数执行
由此得出Java初始化顺序结论:
1 继承体系的所有静态成员初始化(先父类,后子类)
2 父类初始化完成(普通成员的初始化-->构造函数的调用)
3 子类初始化(普通成员-->构造函数)
还有静态嵌套类的初始化,当使用外部类时并不会初始化静态嵌套类,只有使用到内部类时才会初始化内部类。