继承关系中的初始化与加载顺序

先上结论

在使用继承时,就相当于已经知道了基类的一切,并可以访问其中的任何public 和 protected成员

【1.加载基类静态成员及static块执行】

​ 在类的加载过程中,如果该类有基类,编译器会先加载该类的基类。(无论是否创建了基类的对象,基类也会被加载)

因此不会出现 C++中的 一个static 期望使用另一个static ,而另一个static还没被初始化,这就会出现问题

⬇️⬇️⬇️⬇️⬇️

然后基类的static初始化开始执行

【通过输出分析:】初始化X1为null,加载printInit方法,执行X1处的调用

【2.加载派生类静态成员及static块执行】

​ 再者是派生类的static

【3.基类普通成员及始化、从上到下、如果有构造代码块也就执行了】

【4.执行基类构造】

执行基类构造,其对象中的所有基本类型变量先被设置为默认值, 引用类型被设置为null,然后再执行出现在字段定义处的初始化动作(赋值)。

【5.派生类成员初始化、从上到下、如果有构造代码块也就执行了】

【6.执行派生类构造】

结论:Java继承关系中的初始化顺序如下:
父类静态成员、静态代码块 ➡️​ 子类静态成员静态、代码块 ➡️​ 父类成员变量和非静态块(顺序加载) ➡️​ 父类构造函数 ➡️​ 子类成员变量和非静态块(顺序加载) ➡️​ 子类构造函数​

代码验证

继承关系中的执行顺序代码
package onjava8.extend;

/**
 * 继承关系中的执行顺序
 */
class Insect {
    private int i = 9;
    protected int j;

    {
        System.out.println("Insect-构造代码块1执行");
    }
    static  {
        System.out.println("Insect-静态代码块1执行");
    }

    Insect() {
        System.out.println("i = " + i + ", j = " + j);
        j = 39;
    }

    {
        System.out.println("Insect-构造代码块2执行");
    }

    static  {
        System.out.println("Insect-静态代码块2执行");
    }

    private static int x1 = printInit("static Insect.x1 initialized");

    static int printInit(String s) {
        System.out.println(s);
        return 47;
    }
}

public class Beetle extends Insect {

    {
        System.out.println("Beetle-构造代码块1执行");
    }

    private int k = printInit("Beetle.k.initialized");

    static  {
        System.out.println("Beetle-静态代码块1执行");
    }

    public Beetle() {
        System.out.println("k = " + k);
        System.out.println("j = " + j);
    }

    {
        System.out.println("Beetle-构造代码块2执行");
    }

    static  {
        System.out.println("Beetle-静态代码块2执行");
    }
    private static int x2 = printInit("static Beetle.x2 initialized");
}

//测试主函数类
package onjava8.extend;
/**
 * @ClassName BeetleMain
 * @Description TODO
 * @Author fs
 * @Date 2021/12/16 22:22
 * @Version 1.0
 */
public class BeetleMain {
    public static void main(String[] args) {
        System.out.println("Beetle constructor");
        Beetle b = new Beetle();
    }
}
程序输出
Beetle constructor
Insect-静态代码块1执行
Insect-静态代码块2执行
static Insect.x1 initialized
Beetle-静态代码块1执行
Beetle-静态代码块2执行
static Beetle.x2 initialized
Insect-构造代码块1执行
Insect-构造代码块2执行
i = 9, j = 0
Beetle-构造代码块1执行
Beetle.k.initialized
Beetle-构造代码块2执行
k = 47
j = 39

参考连接:https://www.cnblogs.com/blknemo/p/13848051.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
http://blog.csdn.net/methods2011/article/details/8584463 Java继承与静态static等的执行先后顺序的面试题 java面试题静态加载顺序构造方法 继承与static 面试题目如下:请写出程序执行完成之后的结果。 package extend; public class X { Y y=new Y(); static{ System.out.println("tttt"); } X(){ System.out.println("X"); } public static void main(String[] args) { new Z(); } } class Y{ Y(){ System.out.println("Y"); } } class Z extends X{ Y y=new Y(); static{ System.out.println("tt"); } Z(){ System.out.println("Z"); } } 先不告诉最后结果,我们先来分析下。一步一步推出结果。 1.首先分析一段程序的执行后的结果,我们得先找到程序的入口,然后才能着手分析。 也就是main()方法。 2.我们发现main()方法在X类,要执行main()方法,还得先将X类加载到内存。 3.X类加载完成后,会做什么事情呢?别急,先来看看static的作用,不知道吧。告诉你:static就是在类被第一次加载的时候执行,以后就不再执行。 4.知道了static的作用,那么X类被加载,那么就会先执行X类的静态属性和静态语句块(static),执行先后顺序看谁在前面就先执行谁。只在此时执行,以后都不会。 5.所以一个输出结果为tttt,没问题了吧。 6.X类的static语句块执行完了,就该执行main()方法啦。 7.new Z();此方法被执行。 8.既然new Z();那么Z类就要被加载。因为Z类继承X类。所以必须先加载X类才行。因为X类已经被加载。所以此时不用再加载X类了。Z类加载好了就要执行Z类的static语句块 9.那么就会打印出tt了吧。 10.都加在完后就要实例化对象了。 11.实例化Z之前,还得先实例化X对吧。因为子类的构造方法都会调用父类的构造方法。 12.那就先实例化X类吧。 13.执行X方法前还得先初始化对不。也就是获取所有属性。那么X类的属性Y就会获取。 14.即X类的Y y=new Y();要被执行。也就是会打印Y。 15.接着执行System.out.println("X"); 16.然后就是执行Z的构造方法 17.同样先获取Z的属性Y y=new Y();打印Y。 18.再执行System.out.println("Z"); 整个过程就是这样了。现在知道结果了吧: tttt tt Y X Y Z http://snow4909.diandian.com/post/2013-02-17/40049419937 记住,面试有可能会标出一些语句的,让你选择这些语句的执行顺序。不过只要知道原理,就没什么难的了。 执行先后顺序: 1、类加载到内存时static 加载进内润 2、调用类的构造方法时先调用父类的构造方法,在调用子类的构造方法 3、类初始化时,先初始化类的属性成员,在执行构造方法。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值