看代码如下:
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对象时执行,而且在构造器之前执行!其实初始化块就是构造器的补充,初始化快是不能接收任何参数的,定义的一些所有对象共有的属性、方法等内容时就可以用初始化块了初始化!!好处是可以提高初始化块的复用,提高整个应用的可维护性。
最后是构造函数。
最终顺序: 顶层父类的静态块-->父类静态块--->本类静态块----->
-->顶层父类构造代码块 ---->顶层父类构造函数 -->父类构造代码块---->父类构造函数
--->本类.....
