类的加载顺序
创建子类对象的时候
- 父类的静态代码块
- 子类的静态代码块
- 父类代码块
- 父类的构造方法
- 子类代码块
- 子类的构造方法
父类的静态字段——>父类静态代码块——>子类静态字段——>子类静态代码块——>
父类成员变量(非静态字段)——>父类非静态代码块——>父类构造器——>子类成员变量——>子类非静态代码块——>子类构造器
为什么是这个顺序
代码验证
父类代码:
package a3;
public class Father {
public Father() {
System.out.println("父类的构造方法");
}
{
System.out.println("父类代码");
}
static {
System.out.println("父类静态方法");
}
}
子类代码:
package a3;
public class Son extends Father{
public Son( ) {
System.out.println("子类的构造方法");
}
{
System.out.println("子类代码");
}
static {
System.out.println("子类静态方法");
}
public static void main(String[] args) {
Son s =new Son();
}
}
执行结果:
为什么会是这样的结果
原因:
首先static 是静态方法的一个关键字;
而static有几个特性:其中最重要的是:以下两个
- 被static关键字修饰的属性和方法 无须创建对象使用类名可以直接调用
- Static关键字所修饰的内容在整个类执行之前优先被加载
而且这个是在创建对象的时候就进行调用的。
第二个:为什么{}会优先于构造方法:
在执行类加载的时候,jvm不可避免地需要进行初始化
逻辑上
初始化代码块主要用于给类中的局部变量进行初始化的,而构造函数内可能要操作到局部变量,这样必须保证在操作之前所操作的变量得到必要的初始化,因此,初始化代码块必须在构造函数之前执行。构造代码块是随着类的加载而加载的,只要类已存在,构造代码块就加载进内存,而构造函数时在创建对象的时候才会执行,因为此构造代码块会优先于构造函数执行。
三原则
1、父类优先于子类
2、属性优先于代码块优先于构造方法
3、静态优先于非静态
因此,类加载顺序为:
父类静态变量->父类静态语句块->子类静态变量->子类静态语句块->父类普通成员变量->父类动态语句块->父类构造器->子类普通成员变量->子类动态语句块->子类构造器
如果还有疑问:
可以在使用软件的时候启用Ctrl+F8的快捷键进行查看debug进行调试具体是如何生成的