【JAVA】父类和子类包换静态方法、静态变量的加载顺序

128 阅读3分钟

本文已参与「新人创作礼」活动,一起开启掘金创作之路。

题目如下:

思考代码运行结果

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)");  //非静态代码块
    }

    private int test() {
        System.out.println("(4)");
        return 1;
    }

    private static int method() {
        System.out.println("(5)");
        return 1;
    }
}


Son

public class Son extends Father {
    private int i = test();
    private static int j = method();
    static {
        System.out.println("(6)");
    }
    Son(){
        super(); //在子类构造器中一定会调用父类构造器,写或者不写都一样!!!
        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();
        System.out.println();
        Son s2 =new Son();
    }
}

解题所用知识点总结

类的初始化过程

1.一个类要创建实例需要先加载并初始化该,
注意:main 方法所在类需要先加载和初始化
2.一个子类要初始化,一定会先初始化父类
3.一个初始化就是执行<clinit>()方法,这个方法是编译器生成的
    <clinit>()方法由静态类变量显示复制代码和静态代码块组成
    类变量显示赋值代码和静态代码块从上到下顺序执行
    <clinit>()方法只执行一次

实例初始化过程

1.就是执行<init>()方法
       <init>()方法可能重载有多个,有几个构造器就有几个<init>方法
       <init>()方法由非静态实例变量显式赋值代码 和 非静态代码块,
       			对应构造器代码块组成。
       非静态实例变量显式赋值代码块和非静态代码快从上到下的
       			顺序执行,而对应的构造器代码最后执行。
       每次创建实例对象,调用对应的构造器,执行的就是<init>方法
       <init>方法的首行是super()或super(实参列表),即父类的<init>方法 

方法的重写Override

1) 哪些方法不会被重写
    final方法
    静态方法
    private等子类中不可见方法
2) 对象的多态性
    子类如果重写了父类的方法,通过子类对象调用的一定是子类重写过的代码
    非静态方法默认的调用对象是this
    this对象在构造器或者说<init>方法中就是正在创建的对象

解题步骤:

1)

  • 先初始化父类得:(5)(1)
  • 初始化子类得:(10)(6)

2)根据上述知识知首先分析父类

  • 父类< clinit >()
  • j = method();
  • 父类静态代码块
  • 父类实例化方法:
  • (1):super() 最前
  • (2):i = test();
  • (3):父类的非静态代码块
  • (4):父类的无参构造方法 最后
  • 非静态方法前面有一个默认的this
  • this在构造器(或)他表示的是正在创建的对象,因为这里是在创建Son对象,所以 test()执行的是子类重写的代码(面向对象多态)
  • 这里的i=test()执行的是子类重写的test()方法

3)

  • 子类实例化方法:
  • (1):super() 最前 (4)(3)(2)
  • (2):i = test(); (9)
  • (3):子类的非静态代码块 (8)
  • (4):子类的无参构造方法 最后 (7)

4) 最后: 由于创建了两个Son对象,因此实例化方法执行两次

运行结果

(5)
(1)
(10)
(6)
(4)
(3)
(2)
(9)
(8)
(7)

(4)
(3)
(2)
(9)
(8)
(7)