先上代码:
class root{
private int number;
public root(){
number = 2;
System.out.println("root的无参构造器"+number);
}
static {
System.out.println("root的静态代码块");
}
{
System.out.println("root的非静态代码块"+number);
}
}
class leaf extends root{
public leaf() {
// TODO 自动生成的构造函数存根
super();
System.out.println("leaf的无参构造器");
}
static {
System.out.println("leaf的静态代码块");
}
{
System.out.println("leaf的非静态代码块");
}
}
public class test {
public static void main(String[] args) {
new root();
//System.out.println("***************");
//new leaf();
}
}
创建对象后控制台输出如下:
root的静态代码块
root的非静态代码块0
root的无参构造器2
由此我们可以得知,①在同一个类中,三者的执行顺序是 静态代码块》非静态代码块》构造器。
将上述代码中的main方法的注释删除,再次执行,结果如下图:
public class test {
public static void main(String[] args) {
//new root();
//System.out.println("***************");
new leaf();
}
}
结果:
root的静态代码块
leaf的静态代码块
root的非静态代码块0
root的无参构造器2
leaf的非静态代码块
leaf的无参构造器
由此得到结论:②不同静态代码块的执行顺序是由上而下的,从父类到子类,也可认为是书写的先后顺序。
③new leaf()过程中,需要使用类中的一些信息,所以此时开始对类进行加载(只加载这一次),开始执行静态代码块,由于leaf类是继承自root类的,所以先加载父类root,后加载子类leaf,加载同时执行静态代码块。然后,创建对象需要先调用父类的构造器,此时进入root类中,由结论①我们知道代码块会优于构造器执行,当执行完父类root中的非静态代码块和构造方法后,进入子类leaf中再接着执行leaf类的非静态代码块和构造器。
我们再将代码稍作修改进行测试:
public class test {
public static void main(String[] args) {
new root();
System.out.println("***************");
new leaf();
}
}
结果:
root的静态代码块
root的非静态代码块0
root的无参构造器2
***************
leaf的静态代码块
root的非静态代码块0
root的无参构造器2
leaf的非静态代码块
leaf的无参构造器
由此我们得到结论④:静态代码块只会随着类的加载而执行,且只执行一次,而非静态代码块会随着对象的加载而执行,每new一个对象则会执行一次。