如果面试官让你分析类初始化阶段的死锁现象

简介: 哈喽,我是子牙。十余年技术生涯,一路披荆斩棘从技术小白到技术总监到JVM专家到创业。技术栈如汇编、C语言、C++、Windows内核、Linux内核。特别喜欢研究虚拟机底层实现,对JVM有深入研究。分享的文章偏硬核,很硬的那种。

哈喽,我是子牙。十余年技术生涯,一路披荆斩棘从技术小白到技术总监到JVM专家到创业。技术栈如汇编、C语言、C++、Windows内核、Linux内核。特别喜欢研究虚拟机底层实现,对JVM有深入研究。分享的文章偏硬核,很硬的那种。

手撸过JVM、内存池、垃圾回收算法、synchronized、线程池、NIO、三色标记算法…

image.png

准备写两篇文章透彻剖析下类的初始化阶段及初始化阶段的死锁问题:

  1. 类的初始化做什么
  2. JVM底层是如何实现类的初始化的
  3. 为什么会出现死锁问题
  4. 怎么解释死锁问题
  5. 如果证明你对死锁的判断是正确的
  6. 我是如何论证的(改Hotspot源码打日志)

会由浅入深,循序渐进展开。今天是第一篇,难度偏认知层面。读起来应该会很轻松愉快。

clinit

类初始化阶段做什么?其实很简单,执行clinit方法。这个方法哪里来的?你的Java代码中只要有静态属性或者是静态代码段,在编译的时候就会自动生成这个方法。

代码中有静态属性,如图

image.png

代码中有静态方法,如图

image.png

这段代码有两个地方需要注意下:1、静态代码块前可以写static,也可以不写;2、如果代码中有多个静态代码块,编译系统会合并成一个,合并后的代码顺序跟静态代码块的先后顺序保存一致。

死锁现象

国际惯例,上代码

image.png

这段代码运行起来的结果,如图

image.png

代码永远不会结束,其实在JVM层面,是发生了死锁。

检测死锁

一提到死锁,我们会马上想起JVM提供的工具:jconsole、visualvm…期待的结果是

image.png

但是很遗憾,目前没有工具能够检测出初始化阶段发生的死锁问题(我准备尝试写一个)。

为什么jconsole检测不到呢?看图

image.png

线程根据执行代码进入的空间来分,有这四种基本状态,目前能看得到的工具,或者是JVM开放的API,只能做到检测出thread_in_Java这种状态的线程死锁问题。

所以这个问题只能通过阅读Hotspot源码找答案。所以可以推测出面试官问你这个问题,就是看你有没有这个能力,或者是这个习惯,通过阅读Hotspot源码找答案。

初始化流程

类的初始化阶段,对应Hotspot源码:InstanceKlass::initialize_impl。考虑到很多小伙伴不想看C++代码,我用伪代码把意思表达到

讲两个预备知识:

  1. 看这段代码的时候要区别正在执行初始化的线程及其他线程。假设正在执行初始化的线程为T1,又进来一个线程T2执行初始化
  2. Hotspot源码中此处有两个状态对理解代码至关重要:being_initialized(正在执行初始化)、fully_initialized(完成初始化)

image.png

问题的答案已经很清晰了:

  1. 第一个线程执行new指令,触发加载类A,然后执行A的初始化方法clinit,clinit方法中又执行到new指令,触发加载类B,并执行类B的初始化方法clinit
  2. 第二个线程触发加载类B,在类B的clinit方法中又触发加载类A
  3. 死锁的原因就是线程一跟线程二都进入了wait,也就是初始化流程的Step 2
  4. 其实这个问题存在时间差,如果某个线程跑得足够快,完成了初始化,死锁就不会发生。所以如果你的程序出现有时候卡着不动,有时候又是正常的,不妨大胆猜测有可能是发生了初始化阶段死锁。正常来讲,出现死锁的频率更高,我测试了很多次,理想情况发生的次数还是很少很少的

然,目前得出的答案是从理论层面分析出来的,那事实是不是如此呢?如何证明呢?下篇文章分享

我是子牙老师,喜欢钻研底层,深入研究Windows、Linux内核、JVM。如果你也喜欢研究底层,欢迎关注我的公众号【硬核子牙】

相关文章
|
11天前
|
Java 数据库连接 Maven
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)
自动装配是现在面试中常考的一道面试题。本文基于最新的 SpringBoot 3.3.3 版本的源码来分析自动装配的原理,并在文未说明了SpringBoot2和SpringBoot3的自动装配源码中区别,以及面试回答的拿分核心话术。
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)
|
5月前
|
安全 Java 容器
【Java集合类面试二十七】、谈谈CopyOnWriteArrayList的原理
CopyOnWriteArrayList是一种线程安全的ArrayList,通过在写操作时复制新数组来保证线程安全,适用于读多写少的场景,但可能因内存占用和无法保证实时性而有性能问题。
|
5月前
|
Java
【Java集合类面试二十八】、说一说TreeSet和HashSet的区别
HashSet基于哈希表实现,无序且可以有一个null元素;TreeSet基于红黑树实现,支持排序,不允许null元素。
|
5月前
|
Java
【Java集合类面试二十六】、介绍一下ArrayList的数据结构?
ArrayList是基于可动态扩展的数组实现的,支持快速随机访问,但在插入和删除操作时可能需要数组复制而性能较差。
|
2月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
137 4
|
4月前
|
安全 Java 应用服务中间件
JVM常见面试题(三):类加载器,双亲委派模型,类装载的执行过程
什么是类加载器,类加载器有哪些;什么是双亲委派模型,JVM为什么采用双亲委派机制,打破双亲委派机制;类装载的执行过程
109 35
JVM常见面试题(三):类加载器,双亲委派模型,类装载的执行过程
|
2月前
|
SQL 算法 关系型数据库
面试:什么是死锁,如何避免或解决死锁;MySQL中的死锁现象,MySQL死锁如何解决
面试:什么是死锁,死锁产生的四个必要条件,如何避免或解决死锁;数据库锁,锁分类,控制事务;MySQL中的死锁现象,MySQL死锁如何解决
|
3月前
|
JSON 调度 数据库
Android面试之5个Kotlin深度面试题:协程、密封类和高阶函数
本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点。文章详细解析了Kotlin中的协程、扩展函数、高阶函数、密封类及`inline`和`reified`关键字在Android开发中的应用,帮助读者更好地理解和使用这些特性。
43 1
|
5月前
|
存储 Java
【Java集合类面试二十九】、说一说HashSet的底层结构
HashSet的底层结构是基于HashMap实现的,使用一个初始容量为16和负载因子为0.75的HashMap,其中HashSet元素作为HashMap的key,而value是一个静态的PRESENT对象。
|
5月前
|
Java
【Java集合类面试三十】、BlockingQueue中有哪些方法,为什么这样设计?
BlockingQueue设计了四组不同行为方式的方法用于插入、移除和检查元素,以适应不同的业务场景,包括抛异常、返回特定值、阻塞等待和超时等待,以实现高效的线程间通信。

热门文章

最新文章

相关实验场景

更多