JAVA构造器、静态上下文的执行时机与代码执行顺序详解

先看代码案例

package com.exemple.demo.test.preciate;

public class Father {
    private int i = test();
    private static int j = method();

    public Father() {
        System.out.print("(3)");
    }

    static {
        System.out.print("(1)");
    }

    {
        System.out.print("(2)");
    }



    private static int method() {
        System.out.print("(4)");
        return 0;
    }

    public int test() {
        System.out.print("(5)");
        return 0;
    }


}

class Son extends Father{

 
    private int i = test();
    private int k = test();
    private static int j = method();

    static {
        System.out.print("(6)");
    }

    {
        System.out.print("(7)");
    }

    public Son() {
        System.out.print("(8)");
    }

    public Son(int i) {
        this.i = i;
    }

    public Son(int i, int k) {
        this(i);
        this.k = k;
    }

    private static int method() {
        System.out.print("(9)");
        return 0;
    }

    public int test() {
        System.out.print("(10)");
        return 0;
    }

    public static void main(String[] args) {
        Son son = new Son();
        System.out.println();
        Son son2 = new Son();
    }

}

执行结果

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

静态上下文的执行时机

类加载场景

常见的主动加载情况

  • 实例化一个对象时
  • 通过反射加载一个类
  • 调用一个类的静态方法/属性
  • 读写一个类的静态属性
  • 实例化子类对象时(会触发父类的加载)

类加载过程

  • 加载阶段:类加载器ClassLoader从网络或本地文件系统中加载字节码文件到JVM中
  • 链接阶段:将验证Class文件中的字节流包含的信息是否符合当前虚拟机的要求,为静态域分配存储空间并设置类变量的初始值(默认的零值)
  • 初始化:到了此阶段,才真正开始执行类中定义的java程序代码(字节码层面即执行clinit)。clinit即按在代码中出现的顺序执行该类的静态变量初始化和类的静态代码块,如果该类有父类的话,则优先对其父类进行初始化

字节码clinit案例

以Son类的cinit方法为例
在这里插入图片描述

构造器

为了初始化成员属性,而不是初始化对象,初始化对象是通过new关键字实现的,new关键字会在堆上为对象开辟一块空间用来存储对象自己的属性、方法,如果有父类的话也会存储父类相关的非private的属性和方法,因为创建子类对象会调用父类构造方法但不会创建父类对象,只是调用父类构造方法初始化父类成员属性并一块存入到子类对象中

特殊案例

抽象类可以有构造方法,但抽象类不能实例化

字节码init方法

java中的构造器代码会编译为init方法,init方法中的内容首先是执行父类的init方法(子类构造器第一行其实就是super,只不过父类没有多参构造器、或者在有多参构的情况下显示定义了无参构造,那么子类构造器第一行的super()可以省略),随后按代码顺序初始化类属性、普通代码块,(始终)最后执行构造方法,构造方法的执行始终是最后一步去执行的,和代码顺序无关,这意味着构造器赋值拥有最终解释权,实例对象属性最后的值是通过构造器赋予的,一个构造器对应一个init方法,一个类有n个构造器,就会对应n个init方法。

init方法执行过程注意点

init方法按代码顺序初始化类属性的过程中,可能会受到子类重写方法的影响,因为我们在main方法中实例化的是Son类的对象,在实例化Son之前初始化父类Father时,加载父类属性i的时需要执行test方法,看test方法是否被子类给重写了,如果发生了重写,那么此时默认执行的是子类方法中的test,代码执行结果为

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

作为对比,由于我们上述案例中父类中的test方法修饰符是private,所以子类并未发生重写,所以此时执行的还是父类的test,代码执行结果为

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

通过this调用构造器

在这里插入图片描述

init方法案例

以Son类的构造器init方法为例
在这里插入图片描述

  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java中,静态变量、静态代码块和静态方法的执行顺序是按照以下的规则进行的: 1. 首先,父类的静态变量和静态代码块会先执行,按照它们的声明顺序执行。 2. 接着,子类的静态变量和静态代码块会执行,也是按照它们的声明顺序执行。 3. 然后,父类的实例变量和非静态代码块会执行,同样按照它们的声明顺序执行。 4. 紧接着,父类的构造方法会执行。 5. 然后,子类的实例变量和非静态代码块会执行,也是按照它们的声明顺序执行。 6. 最后,子类的构造方法会执行。 需要注意的是: 1. 非静态代码块会在构造方法之前执行,并且每次实例化对象之前都会先执行静态代码块。而静态代码块会在非静态代码块之前执行。 2. 静态代码块只会在类第一次被调用时执行一次。 3. 静态代码块只能调用静态变量和静态方法,而非静态代码块或非静态方法可以调用任何(静态或非静态)变量。 总结起来,静态变量、静态代码块和静态方法的执行顺序是先父类的静态变量和静态代码块,再子类的静态变量和静态代码块,然后父类的实例变量和非静态代码块,接着是父类的构造方法,然后子类的实例变量和非静态代码块,最后是子类的构造方法。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [java学习之静态变量、静态语句块、静态方法、语句块的执行顺序](https://blog.csdn.net/pig_small/article/details/79975158)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* [Java静态)变量、(静态代码块、构造方法的执行顺序](https://blog.csdn.net/weixin_30823683/article/details/97356747)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值