首先需要知道,一个类里面可以有多个初始化块,相同类型的初始化块之间有顺序:前面定义的初始化块先执行,后面定义的初始化块后执行。
初始化块的修饰符只能是static,使用static修饰的初始化块被称为类初始化块(静态初始化块),没有static修饰的初始化块被称为实例初始化块(非静态初始化块)。
实例初始化块和构造器
实例初始化块的基本用法:如果有一段初始化处理代码对所有对象完全相同,且无须接收任何参数,就可以把这段初始化处理代码提取到实例初始化块。
实际上实例初始化块是一个假象,使用javac命令编译Java类后,该Java类中的实例化代码块会消失——实例化代码块中代码会被“还原”到每个构造器中,且位于构造器所有代码的前面
类初始化块
如果定义初始化块时使用了static修饰符,则这个初始化块就变成了类初始化块,也被称为静态初始化块(实例初始化块负责对对象执行初始化,类初始化块则负责对类进行初始化)。类初始化块是类相关的,系统将在类初始化阶段执行类初始化块,而不是在创建对象时才执行。因此类初始化总是比实例初始化块先执行。
下面的代码创建了三个类:Root、Mid和Leaf,这三个类都提供了类初始化块和实例初始化块,而且Mid类里面还使用this调用重载的构造器,而Leaf使用super显示调用其父类指定的构造器。
class Root{
static
{
System.out.println("Root的类初始化块");
}
{
System.out.println("Root 的实例初始化块");
}
public Root() {
System.out.println("Root 的无参数的构造器");
}
}
class Mid extends Root{
static {
System.out.println("Mid的类初始化块");
}
{
System.out.println("Mid 的实例初始化块");
}
public Mid() {
System.out.println("Mid 的无参数的构造器");
}
public Mid(String msg) {
// 通过this调用同一类中重载的构造器
this();
System.out.println("Mid 的带参数构造器,其参数值:"+
msg);
}
}
class Leaf extends Mid{
static {
System.out.println("Leaf的类初始化块");
}
{
System.out.println("Leaf 的实例初始化块");
}
public Leaf() {
// 通过super调用父类中有一个字符串参数的构造器
super("疯狂java讲义");
System.out.println("执行Leaf的构造器");
}
}
public class Test {
public static void main(String[] args) {
// TODO Auto-generated method stub
new Leaf();
new Leaf();
}
}
三个类的继承结构
从图中可以看出,第一次创建一个Leaf对象时,因为系统中还不存在Leaf类,因此需要先加载并初始化Leaf类,初始化Leaf类时会先执行其顶层父类的类初始化块,再执行期直接父类的类初始化块,最后才执行Leaf本身的类初始化块。
一旦Leaf类初始化成功后,Leaf类在该虚拟机里将一直存在,因此当第二次创建Leaf实例时无须再次对Leaf类进行初始化。
注意:Java系统加载并初始化某个类时,总是保证该类的所有父类(包括直接父类和间接父类)全部加载并初始化。