public class Parents {
//静态代码块 随着类的加载而执行,而且只执行一次
//作用:初始化类的信息
//如果一个类中定义了多个静态代码块,则按照声明的先后顺序执行
//非静态代码块(实例化代码块) 内部可以有输出语句
//对着对象的创建而执行
//每创建一次对象,就执行一次非静态代码块
static {
System.out.println("父的静态代码块");
}
{
System.out.println("父的实例代码块");
}
public Parents(){
System.out.println("父的构造方法");
}
}
public class Son extends Parents{
static {
System.out.println("子的静态代码块");
}
{
System.out.println("子的实例代码块");
}
public Son(){
System.out.println("子的构造方法");
}
}
public class Test {
public static void main(String[] args) {
//代码块一定比构造方法先执行 代码块可以处理一些逻辑
//类的加载过程分为主要分为五步:1、加载;2、验证;3、准备;4、解析;5、初始化。
//遇到new,putstatic,getstatic,invokestatic这四条字节码指令时,如果该类没有进行过初始化,则会触发初始化;生成这四条指令的最常见的java代码场景是:使用new关键字实例化对象的时候、读取或者设置一个类的静态字段(不包括静态常量)的时候,以及调用一个静态方法的时候
//静态代码块的加载时机是在类加载过程中的第五步中完成的,也就是类加载过程中的初始化阶段。
//Parents parents = new Son();
Son son = new Son();
}
//输出结果
//父的静态代码块
//子的静态代码块
//父的实例代码块
//父的构造方法
//子的实例代码块
//子的构造方法
}
个人理解:
当new son()的时候实际就是调用son的无参构造函数,默认的无参构造函数就会调用super(),调用父类的构造方法,因为父类有静态代码块,静态代码块是在初始化的时候就会调用,所以首先是父类静态代码块,然后是子类的静态代码块,首先是调用父类的构造方法,肯定是先创建父类,又因为父类有实例化代码块,所以父类的实例化代码块先执行,然后再是父类的构造方法,子类的实例化代码,子类的构造方法