代码示例:
情况一:没有继承父类时
class HelloA { public HelloA() { System.out.println("I'm A class"); } static { System.out.println("static A");//特点:随着类的加载而执行,且只执行一次,并优先于主函数。用于给类初始化的。 } { System.out.println("A"); } public static void main(String[] args) { new HelloA(); new HelloA(); } }
输出结果:
static A
A
I'm A class
A
I'm A class
分析:输出顺序为: 静态代码块>非静态代码块>构造器,同时无论New多少次class静态代码块只执行一次,所以一般情况把耗内存重量级的连接或者其他放在静态代码块中,访问速度快,并且只访问一次,节约消耗。
情况二:继承某个父类时
class HelloA extends HelloB{ public HelloA() { System.out.println("I'm A class"); } static { System.out.println("static A"); } { System.out.println("A"); } public static void main(String[] args) { new HelloA(); } } class HelloB { public HelloB() { System.out.println("I'm B class"); } static { System.out.println("static B"); } { System.out.println("B"); } }
运行结果:
static B
static A
B
I'm B class
A
I'm A class
分析:输出顺序:父类静态代码块>子类静态代码块>父类非静态代码块>父类构造器>子类非静态代码块>子类构造器;
区别:
静态代码块,在虚拟机加载类的时候就会加载执行,而且只执行一次;
非静态代码块,在创建对象的时候(即new一个对象的时候)执行,每次创建对象都会执行一次
分析:对象的初始化顺序:首先执行父类静态的内容,父类静态的内容执行完毕后,接着去执行子类的静态的内容,当子类的静态内容执行完毕之后,再去看父类有没有非静态代码块,如果有就执行父类的非静态代码块,父类的非静态代码块执行完毕,接着执行父类的构造方法;父类的构造方法执行完毕之后,它接着去看子类有没有非静态代码块,如果有就执行子类的非静态代码块。子类的非静态代码块执行完毕再去执行子类的构造方法。总之一句话,静态代码块内容先执行,接着执行父类非静态代码块和构造方法,然后执行子类非静态代码块和构造方法。
而且子类的构造方法,不管这个构造方法带不带参数,默认的它都会先去寻找父类的不带参数的构造方法。如果父类没有不带参数的构造方法,那么子类必须用supper关键子来调用父类带参数的构造方法,否则编译不能通过。