Java对象初始化过程代码块与构造器调用顺序

前言

对Java对象初始化过程 代码块与构造器调用顺序进行整理说明。先说结论具体论证在下文。

代码加载的优先级顺序

静态代码块、静态成员变量->非静态代码块、非静态成员变量->new其他对象调用对应对象构造方法(在本地对象的方法外包括构造方法)->new本地对象调用构造方法。

【注意:若new对象时,该对象中有静态代码块和非静态代码块,每new一次对象,非静态代码块都会执行一次,但静态代码块只会执行一次往后new对象都不会再执行。】

构造方法的执行顺序

父类静态代码块(静态变量 > 静态块) > 子类的静态代码块 > 父类构造代码块、构造方法> 子类的构造代码块、构造方法

各种代码块的定义

静态代码块

class Demo{ static { //静态代码块...... } }

特点: 1、Java静态代码块中的代码会在类加载JVM时运行,且只被执行一次 2、静态块常用来执行类属性的初始化 ,和一些全局初始化的工作 3、静态块优先于各种代码块以及构造函数,如果一个类中有多个静态代码块,会按照书写顺序依次执行 4、静态代码块可以定义在类的任何地方中除了方法体中【这里的方法体是任何方法体】 5、静态代码块不能访问普通变量

有关静态代码块再详细介绍下:

静态代码块:在java中使用static关键字声明的代码块。静态块用于初始化类,为类的属性初始化。每个静态代码块只会执行一次。

由于JVM在加载类时会执行静态代码块,所以静态代码块先于主方法执行。 如果类中包含多个静态代码块,那么将按照"先定义的代码先执行,后定义的代码后执行"。 【注意:1 静态代码块不能存在于任何方法体内。2 静态代码块不能直接访问静态实例变量和实例方法,需要通过类的实例对象来访问。】 实例代码块 实例代码块 又叫 构造初始化块 , 构造代码块 , 初始化块 。

class Demo{ { //实例代码块...... } }

特点:

1、构造代码块在创建对象时被调用,每次创建对象都会调用一次
2、构造代码块优先于构造函数执行,同时构造代码块的运行依赖于构造函数
3、构造代码块在类中定义

局部代码块

局部代码块又叫 普通代码块 , 方法代码块

class Demo{ public void test(){ { //局部代码块...... } } }

特点: 1、普通代码块定义在方法体中 2、普通代码块与实例代码块的格式一致都是{} 3、普通代码块与构造代码块唯一能直接看出的区别是构造代码块是在类中定义的,而普通代码块是在方法体中定义的 4、可以限定变量生命周期,及早释放,提高内存利用率

验证各代码块的执行顺序

举例代码如下:

class Init { public Init() {
        System.out.println("无参构造器");
    } public Init(int a) {
        System.out.println("有参构造器");

    }

    {
        System.out.println("实例代码块1");
    }

    {
        System.out.println("实例代码块2");
    }

    {

        System.out.println("实例代码块3");
    } static {
        System.out.println("静态初始化块1");
    } static {
        System.out.println("静态初始化块2");
    } public void method(){
        {
            System.out.println("普通初始化块");
        }
    }
}

测试代码 如下:

class Demo { public static void main(String[] args) {
        Init init1 = new Init();
        init1.method();
        System.out.println("------------");
        Init init2 = new Init();
        init2.method(); //多打印几个对象的目的是:方便看出Static静态代码块 是否只执行一次!!! System.out.println("------------");
        Init init3 = new Init();
        init3.method();
    }
}

运行结果如下图:


结论:

执行顺序为:静态代码块 > 实例代码块 > 构造函数 > 普通代码块,

且静态代码块,类加载的时候就会调用,且只调用一次(随着类的加载而执行)。

那么类什么时候会被加载呢?

  • 创建对象实例时(new)
  • 创建子类对象实例,父类也会被加载
  • 使用类的静态成员时(静态属性,静态方法)

验证存在继承关系中各代码块的执行顺序

举例继承关系为 Three——> Two——> One, 代码如下:

class One { public One() {
        System.out.println("One构造器");
    }

    {
        System.out.println("One实例化块");
    } static {
        System.out.println("One静态代码块");

    }

} class Two extends One { public Two() {
        System.out.println("Two构造器");
    }

    {
        System.out.println("Two实例化块");
    } static {
        System.out.println("Two静态代码块");
    }

} class Three extends Two { public Three() {
        System.out.println("Three构造器");
    }

    {
        System.out.println("Three实例化块");
    } static {
        System.out.println("Three静态代码块");
    }

} //测试代码 如下: public class Demo { public static void main(String[] args) {
        Three three = new Three();
        System.out.println("-----");
        Three three1 = new Three(); //重复执行的目的是为了 验证static是否只执行一次 System.out.println("-----");
        Two three2 = new Three(); //验证 多态的情况下 用后面的类进行初始化 结果和上面一样 }
}

根据执行结果可知,在多个类的继承中存在初始化块、静态初始化块、构造器,执行真实顺序为:先后执行父类A的静态块,父类B的静态块,最后子类的静态块,然后再执行父类A的实例代码块和构造器,然后是B类的实例代码块和构造器,最后执行子类C的实例代码块和构造器【注:这里的ABC对应One、Two、Three 】

结论:

多个类的继承中初始化块、静态初始化块、构造器的执行顺序为:

父类静态块——>子类静态块——>父类实例代码块——>父类构造器——>子类实例代码块——>子类构造器 ——>(如果有局部代码块, 再正常执行即可, 这里就没必要进行测试了)

通过字节码深究实例代码块优先于构造器原因

我们那一段代码作为例子说明下,代码如下:

class Init { public Init() {
        System.out.println("无参构造器");
    } public Init(int a) {
        System.out.println("有参构造器");

    }

    {
        System.out.println("实例代码块1");
    }

    {
        System.out.println("实例代码块2");
    }

    {
        System.out.println("实例代码块3");
    } static {
        System.out.println("静态初始化块1");
    } static {
        System.out.println("静态初始化块2");
    } public void method(){
        {
            System.out.println("普通初始化块");
        }
    }
}

接下来让我们看看 , Init.java编译完的的字节码文件(Init.class)


从这个字节码文件就可以很清晰的看出, 实例代码块实际上是被依次放到了构造方法的第一句, 所以可以的出此结论: 实例代码块的执行顺序是优先于构造器的。

#java#
全部评论
知识是快乐的源泉
点赞 回复 分享
发布于 2022-08-28 18:02 河南

相关推荐

昨天 16:42
井冈山大学 Java
点赞 评论 收藏
分享
来,说点可能被同行“骂”的大实话。🙊当初接数字马力Offer时,朋友都说:“蚂蚁的“内包”公司?你想清楚啊!”但入职快一年后的今天,我反而对他有了不一样的看法!🔹 是偏见?还是信息差!之前没入职之前外面都在说什么岗位低人一等这类。实际上:这种情况不可至否,不能保证每个团队都是其乐融融。但我在的部门以及我了解的周边同事都还是十分好相处的~和蚂蚁师兄师姐之间也经常开一些小玩笑。总之:身份是蚂蚁公司给的,地位是自己挣的(一个傲娇女孩的自述)。🔹 待遇?玩的就是真实!试用期工资全额发!六点下班跑得快(早9晚6或者早10晚7,动态打卡),公积金顶格交。别听那些画饼的,到手的钱和下班的时间才是真的(都是牛马何必难为牛马)。🔹 能不能学到技术?来了就“后悔”!我们拥有权限直通蚂蚁知识库,技术栈多到学不完。说“学不到东西”的人,来了可能后悔——后悔来晚了(哈哈哈哈,可以不学但是不能没有)!💥 内推地址:https://app.mokahr.com/su/ueoyhg❗我的内推码:NTA6Nvs走我的内推,可以直达业务部门,面试流程更快速,进度可查!今天新放HC,之前挂过也能再战!秋招已经正式开始啦~机会就摆在这,敢不敢来试一试呢?(和我一样,做个勇敢的女孩)
下午吃泡馍:数字马力的薪资一般哇,5年经验的java/测试就给人一万出头,而且刚入职第三天就让人出差,而且是出半年
帮你内推|数字马力 校招
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务