首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

从 static 关键字深入理解 java 对象初始化顺序

前言

最近在阅读 源码的时候,发现一段很有意思的代码,代码片段如下:

以上代码片段主要是 生成哈希值()的逻辑,通过静态的原子整型变量 以及静态方法 ,为每个线程持有的 本地变量生成唯一 的 。

注: 的 选择 变量值: 很有意思,里面涉及到和,感兴趣的同学可以自行了解下。

当然本文的重点不是 原理分析上,而是分析 关键字修饰的静态域(静态变量、静态块)顺序加载问题。

这段代码总共四行,除了第一行都是用 关键字修饰的,这里我们设想一个问题,当类初始化的时候,这四行代码是从上往下执行的吗?

答案是:”否“。

静态变量

为了方便 调试,我们把上面的代码稍微做了下调整,代码片段如下:

上面的代码片段用 模式启动,通过为每行代码打断点,发现当真正实例化 类时,代码运行顺序并非是按照逐行执行,而是如下图红色标记顺序进行的。

其执行流程是:

第一步、用 关键字初始化 类的构造方法

第二步、初始化静态变量

第三步、初始化静态变量

第四步、初始化成员变量

最后 、在 构造方法打印 变量的 值

对象实例化

就是执行类中构造函数的内容,如果该类存在父类,会通过显示或者隐示的方式(方法)先执行父类的构造函数,在堆内存中为父类的实例变量开辟空间,并赋予默认的初始值,然后在根据构造函数的代码内容将真正的值赋予实例变量本身,然后,引用变量获取对象的首地址,通过操作对象来调用实例变量和方法

从上面代码执行流程可以看出

在对象实例化之前必须先初始化 修饰的静态变量,并且静态变量也是有加载顺序的;

类的成员变量的初始化在构造方法里面进行,加载顺序优先于构造方法体的执行语句。

如果某类继承了父类,那么必须先初始化父类的构造方法以及成员变量以及构造方法的执行语句,然后才是子类的成员变量以及构造方法的执行语句。

另外,静态语句块中只能访问到定义在静态块之前的变量,在静态块里可以给该变量赋值,但是不能访问,否则编译器会提示 “” 错误,如下图

静态块

静态块主要用于类的初始化,不是指对象的实例化。它只会执行一次,静态块只能访问类的静态成员属性和方法,不能在静态块使用 。

我们先把上面的代码稍加改造下,增加 “”和“” 静态块代码

运行结果如下:

发现不管是静态块还是静态变量,它们之间都是按顺序执行的。那为什么是静态块、静态变量的初始化是有顺序的呢?

通过查看 类的 编译文件,发现编译器会把 块的代码放在同一 花括号内。

代码顺序是按照之前编码的顺序整合,这么看来是编译器在作怪吧。

类加载中,静态域的加载时机

从《深入理解Java虚拟机:JVM高级特性与最佳实践(第2版)》这本书讲的类加载机制原理可知:

当遇到、 和 或 这条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。

这就解释了为什么在对象未实例化前,可以通过 “类名.静态属性变量、类名.静态方法” 的方式访问静态变量和静态方法了。

类加载的时机

对于初始化阶段,虚拟机规范规定了有且只有 5 种情况必须立即对类进行“初始化”(而加载、验证、准备自然需要在此之前开始):

遇到new、getstatic 和 putstatic 或 invokestatic 这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。对应场景是:使用 new 实例化对象、读取或设置一个类的静态字段(被 final 修饰、已在编译期把结果放入常量池的静态字段除外)、以及调用一个类的静态方法。

对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。

当初始化类的父类还没有进行过初始化,则需要先触发其父类的初始化。(而一个接口在初始化时,并不要求其父接口全部都完成了初始化)

虚拟机启动时,用户需要指定一个要执行的主类(包含 main() 方法的那个类),虚拟机会先初始化这个主类。

当使用 JDK 1.7 的动态语言支持时,如果一个 java.lang.invoke.MethodHandle 实例最后的解析结果 REF_getStatic、REF_putStatic、REF_invokeStatic 的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化。

总结

1、静态域(静态变量、静态块)是按逐行顺序加载的,并且静态域只会加载一次。

2、当实例化对象之前(构造方法调用),会先去初始化静态域,再去调用构造函数实例化对象。

3、一般对象初始化顺序如下:父类的静态域顺序加载–>子类静态域顺序加载–>父类非静态域初始化->父类构造函数初始化–>子类非静态域初始化->子类构造函数初始化。

参考

资料整理

关于我

作者简介:,一枚简单的北漂程序员。喜欢用简单的文字记录工作与生活中的点点滴滴,愿与你一起分享程序员灵魂深处真正的内心独白。我的微信号,输入  ,有份惊喜送给你哦。

如果您觉得本文对你有帮助,欢迎老铁们帮忙

点赞、在看、留言、分享

,你们的支持是我原创最大的动力。

【猿芯】

喜欢就点个"在看"呗^_^

  • 发表于:
  • 原文链接https://kuaibao.qq.com/s/20201123A07BX500?refer=cp_1026
  • 腾讯「腾讯云开发者社区」是腾讯内容开放平台帐号(企鹅号)传播渠道之一,根据《腾讯内容开放平台服务协议》转载发布内容。
  • 如有侵权,请联系 cloudcommunity@tencent.com 删除。

扫码

添加站长 进交流群

领取专属 10元无门槛券

私享最新 技术干货

扫码加入开发者社群
领券