类初始化过程
①一个类要创建实例需要先加载并初始化该类
- main方法所在的类需要先加载和初始化
②一个子类要初始化需要先初始化父类
③一个类初始化就是执行<clinit>()方法
- <clinit>()方法由静态类变量显式赋值代码和静态代码块组成
- 类变量显式赋值代码和静态代码块代码从上到下顺序执行
- <clinit>()方法只执行一次
实例初始化过程
①实例初始化就是执行<init>()方法
- <init>()方法可能重载有多个,有几个构造器就有几个<init>方法
- <init>()方法由非静态实例变量显示赋值代码和非静态代码块、对应构造器代码组成
- 非静态实例变量显示赋值代码和非静态代码块代码从上到下顺序执行,而对应构造器的代码最后执行
- 每次创建实例对象,调用对应构造器,执行的就是对应的<init>方法
- <init>方法的首行是super()或super(实参列表),即对应父类的<init>方法
方法的重写Override
①哪些方法不可以被重写
- final方法
- 静态方法
- private等子类中不可见方法
②对象的多态性
- 子类如果重写了父类的方法,通过子类对象调用的一定是子类重写过的代码
- 非静态方法默认的调用对象是this
- this对象在构造器或者说<init>方法中就是正在创建的对象
/**
* 父类
* @author BarryLee
* @2018年11月10日@下午3:52:33
*/
public class Father {
private int i = test();
private static int j = method();
static {
System.out.print("(1)");
}
public Father() {
System.out.print("(2)");
}
{
System.out.print("(3)");
}
public int test() {
System.out.print("(4)");
return 1;
}
public static int method() {
System.out.print("(5)");
return 1;
}
}
/**
* 子类
* @author BarryLee
* @2018年11月10日@下午3:55:23
*/
public class Son extends Father{
private int i = test();
private static int j = method();
static {
System.out.print("[6]");
}
public Son() {
System.out.print("[7]");
}
{
System.out.print("[8]");
}
public int test() {
System.out.print("[9]");
return 1;
}
public static int method() {
System.out.print("[10]");
return 1;
}
///----测试1----
/*
* 测试类初始化
*/
@Test
public void test1() {
// (5)(1)[10][6][9](3)(2)[9][8][7]
// 相比main多了[9](3)(2)[9][8][7]是因为@Test是类反射,调用了空参构造方法
}
/*public static void main(String[] args) {
// (5)(1)[10][6]
}*/
///----测试2----
/*
* 测试实例初始化、方法重写
*/
@Test
public void test2() {
Son son1 = new Son();
System.out.println("\r\n----");
Son son2 = new Son();
/*
(5)(1)[10][6][9](3)(2)[9][8][7][9](3)(2)[9][8][7]
----
[9](3)(2)[9][8][7]
*/
}
public static void main(String[] args) {
Son son1 = new Son();
System.out.println("\r\n----");
Son son2 = new Son();
/*
(5)(1)[10][6][9](3)(2)[9][8][7]
----
[9](3)(2)[9][8][7]
*/
}
}
最后总结一下:
- 父类静态-子类静态-父类成员变量-父类普通代码块-父类构造-子类成员变量-子类普通代码块-子类构造
- 先静态,再非静态,先父类,再子类
- 上边所说的的 ”静态“ 包括了成员变量和静态代码块
- 非静态的方法还会存在覆盖的情况