类初始化和实例化过程
类初始化过程
1、一个类要创建实例需要先加载并初始化该类
main方法所在的类需要先加载和初始化
2、一个子类要初始化需要先初始化父类
3、一个类初始化就是执行clinit()方法
clinit每个类都有,自动生成 中文名类初始化方法
clinit方法由静态类变量显示赋值代码和静态代码块组成
类变量显示赋值代码和静态代码块代码从上到下顺序执行(谁在上面谁先执行)
clinit方法只执行一次
实例初始化过程
实例初始化就是执行init方法
init方法可能重载由多个,有几个构造器就有几个init方法
init方法由非静态实例变量显示赋值代码块和非静态代码块、对应构造器代码组成
非静态实例变量显示赋值代码和非静态代码块从上到下顺序执行,而对应构造器代码最后执行
每次创建实例对象,调用对应构造器,执行的就是对应的init方法
init方法的首行是super()或super(实参列表),即对应父类的init方法
非静态方法前面其实有一个默认对象this,this在构造器或init他表示正在创建的对象,正在创建的对象是谁,就执行谁的重写代码(面向对象多态)这里创建的是son对象,所以test()执行的就是son的重写方法
方法的重写Override
1、哪些方法不可以被重写
final方法
静态方法
private等子类中不可见方法
2、对象的多态性
子类如果重写了父类的方法,通过子类对象调用的一定是子类重写过的代码
非静态方法默认调用的对象是this
this对象在构造器或者说init方法中就是正在创建的对象
首次运行并实例化
注:类变量和静态代码块,谁的代码在前面就先执行谁,注意子类是否重写了父类方法
初始化父类类变量-->父类静态代码块-->子类类变量-->子类静态代码块-->父类非静态实例变量显示赋值代码-->父类非静态代码块-->父类构造方法-->子类非静态实例变量显示赋值代码-->子类非静态代码块-->子类构造方法
二次实例化
注:非静态实例变量显示赋值代码和非静态代码块从上到下顺序执行,而对应构造器代码最后执行,注意子类是否重写了父类方法
子类非静态实例变量显示赋值代码-->父类静态域-->父类构造方法-->子类非静态实例变量显示赋值代码-->子类静态域-->子类构造方法
示例代码
父类
/**
* @ClassName: Father
* @Description: 父类
* @Author 超霸
* @Date: 2021/1/24 14:01
*/
public class Father {
private int i = test();
private static int j = method();
static {
System.out.print("(1)");
}
Father() {
System.out.print("(2)");
}
{
System.out.print("(3)");
}
public int test() {
System.out.print("(4)");
return 1;
}
private static int method() {
System.out.print("(5)");
return 1;
}
}
子类
/**
* @ClassName: Son
* @Description: 子类
* @Author 超霸
* @Date: 2021/1/24 14:02
*/
public class Son extends Father {
private int i = test();
private static int j = method();
static {
System.out.print("(6)");
}
Son() {
System.out.print("(7)");
}
{
System.out.print("(8)");
}
@Override
public int test() {
System.out.print("(9)");
return 1;
}
private static int method() {
System.out.print("(10)");
return 1;
}
public static void main(String[] args) {
Son s1 = new Son();
System.out.println();
Son s2 = new Son();
}
}
运行结果
(5)(1)(10)(6)(9)(3)(2)(9)(8)(7)
(9)(3)(2)(9)(8)(7)