看代码如下:
Father 父类
Son extends Father
InitClass extends Son
public class InitClass extends Son{
static{
System.out.println("InitClass 静态块....");
}
{
System.out.println("InitClass 构造代码块init....");
}
InitClass(){
System.out.println("InitClass 构造函数初始化");
}
public static void main(String[] args) {
InitClass ic ;
System.out.println("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~实例化分割线~~~~~~~~~~~~~~~~~~~~~~~~");
ic = new InitClass();
}
}
class Father{
static{
System.out.println("Father 静态块....");
}
{
System.out.println("Father 构造代码块init....");
}
Father(){
System.out.println("Father 构造函数初始化");
}
}
class Son extends Father{
static{
System.out.println("Son 静态块....");
}
{
System.out.println("Son 构造代码块init....");
}
Son(){
System.out.println("Son 构造函数初始化");
}
}
输出结果:
Father 静态块....
Son 静态块....
InitClass 静态块....
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~实例化分割线~~~~~~~~~~~~~~~~~~~~~~~~
Father 构造代码块init....
Father 构造函数初始化
Son 构造代码块init....
Son 构造函数初始化
InitClass 构造代码块init....
InitClass 构造函数初始化
可以看出,类在加载但是没有实例化的时候会先加载其最顶层的父类的静态块 staic{ },其次父类的静态块,最后自己的静态块,静态块只加载一次,不论该类被实例化多少次
构造代码块: { } 这里面的代码在创建java对象时执行,而且在构造器之前执行!其实初始化块就是构造器的补充,初始化快是不能接收任何参数的,定义的一些所有对象共有的属性、方法等内容时就可以用初始化块了初始化!!好处是可以提高初始化块的复用,提高整个应用的可维护性。
最后是构造函数。
最终顺序: 顶层父类的静态块-->父类静态块--->本类静态块----->
-->顶层父类构造代码块 ---->顶层父类构造函数 -->父类构造代码块---->父类构造函数
--->本类.....