类初始化过程
- 一个类要创建实例需要先加载并初始化该类
1.1mian方法所在的类需要先加载和初始化 - 一个子类要初始化需要先初始化父类
- 一个类初始化就是执行
<clinit>()
方法
3.1<clinit>()
方法由静态类变量显示赋值代码和静态代码块组成
3.2 类变量显示赋值代码和静态代码块代码从上到下顺序执行
3.3<clinit>()
方法只执行一次
父类Father
public class Father {
private int i = test();
private static int j = method();
static {
System.out.println("(1)");
}
Father() {
System.out.println("(2)");
}
{
System.out.println("(3)");
}
public int test() {
System.out.println("(4)");
return 1;
}
public static int method() {
System.out.println("(5)");
return 1;
}
}
子类Son继承父类Father
public class Son extends Father{
private int i = test();
private static int j = method();
static {
System.out.println("(6)");
}
Son() {
System.out.println("(7)");
}
{
System.out.println("(8)");
}
public int test() {
System.out.println("(9)");
return 1;
}
public static int method() {
System.out.println("(10)");
return 1;
}
public static void main(String[] args) {
Son s1 = new Son();
}
}
当main方法中没有实例化时
public static void main(String[] args) {
}
运行的结果是
(5)
(1)
(10)
(6)
类变量显示赋值代码和静态代码块代码从上到下顺序执行,谁在前先执行谁,所以运行的顺序是
父类初始化<clinit>
(1) j = method() -->(5)
(2) 父类的静态代码块 -->(1)
子类初始化<clinit>
(1) j = method() -->(10)
(2) 子类的静态代码块 -->(6)
实例初始化过程
- 实例初始化就是执行
<init>()
方法<init>()
方法可能重载有多个,有几个构造器就有几个<init>()
方法<init>()
方法由非静态实例变量显示赋值代码和非静态代码块、对应构造器代码组成- 非静态实例变量显示赋值代码和非静态代码块从上到下顺序执行,而对应构造器的代码最后执行
- 每次创建实例对象,调用对应构造器,执行的就是对应的
<init>
方法 <init>
方法的首行是super()
或super
(实参列表),即对应父类的<init>
方法
当main方法中有实例化时
public static void main(String[] args) {
Son s1 = new Son();
}
运行的结果是
(5)
(1)
(10)
(6)
(9)
(3)
(2)
(9)
(8)
(7)
在构造器中一定有super()
方法,无论写或不写都存在,这样的话在子类构造器中一定会调用父类的构造器
运行顺序是
子类实例化
(1) super()(最前)
(1.1) super()(最前)
(1.2) i = test()
(1.2.1) 非静态方法前面其实有一个默认的对象this,this在构造器(或<init>)它表示的是正在创建的对象,
因为这里是在创建Son对象,所以test()执行的是子类重写的代码(面向对象多态)
这里的 i = test()执行的是子类重写的test()方法
(1.3) 父类的非静态代码块
(1.4) 父类的无参构造(最后)
(2) i = test()
(3) 子类的非静态代码块
(4) 子类的无参构造(最后)
注意事项
- 哪些方法不可以被重写
- final方法
- 静态方法
- private等子类中不可见方法
- 对象的多态性
- 子类如果重写了父类的方法,通过子类对象调用的一定是子类重写过的代码
- 非静态方法默认的调用对象是this
- this对象在构造器或者说
<init>
方法中就是正在创建的对象