分享一个新出炉的JVM里不痛不痒的BUG(Attach机制相关)

文章探讨了在并发环境下,如何由于信号处理机制导致JVM中的AttachListener线程可能出现重复创建,尤其是在多个jstack命令同时执行时。作者通过代码修改展示了问题复现的过程,并强调了及时设置关键标记的重要性,以防止线程过度创建。
摘要由CSDN通过智能技术生成

本文来自: PerfMa技术社区

PerfMa(笨马网络)官网

概述

老早之前写过一篇文章,关于attach机制的,可以看下这篇老文章了解一下JVM源码分析之Attach机制实现完全解读,比如大家常用的jstack,jmap等工具的主要原理都和attach机制有关,在JVM里处理这些命令的线程主要是Attach Listener这个线程,这个线程在JVM里是唯一的,我之前也一直以为是唯一的,但是我们同事最近在做一个线程分析产品的时候,发现我们抓到了多个Attach Listener线程,这让我也很疑惑,我第一感觉是不可能,肯定是数据抓错了,直到亲眼看到了两个同名的Attach Listener线程我才不得不相信原来还真有这种情况。

问题分析

不过从Attach Listener的实现来看,它设计的初衷不应该是一个多线程的设计,于是我昨晚上又翻了一遍代码,发现还真可能存在这种情况。举个栗子,当我们很多人同时执行jstack的时候,就可能会发生,当然有个前提是之前都没有做过任何和attach相关的操作。

Attach Listener线程默认情况下不会在JVM启动的时候就创建,当然也有一个JVM参数可以指定在JVM启动的时候就启动这个线程,这个就不会存在我们今天讨论的这个问题了,这个JVM参数是-XX:+StartAttachListener

当我们在运行时触发attach机制的时候,首先会通过Signal Dispatcher线程来创建Attach Listener线程,代码如下:

在上面的圈起来的init方法里会创建Attach Listener线程,但是在init方法执行之前会通过_initialized属性来判断是否需要创建线程,而_initialized设置为true是在attach_listener_thread_entry里,这个是Attach Listener Thread的entry,也就是当这个线程执行的时候执行的方法。

但是在设置_initialized=true之前,如果有多个请求信号发出了(比如同时又很多jstack命令触发),可能会创建多个Attach Listener,因为Signal DispatcherAttach Listener线程是异步执行的。

问题复现

为了让效果更明显,我们可以在hotspot里修改下代码重新编译下再跑demo

在上面函数里加上圈起来的这段代码,表示在设置_initialized属性之前停留15s,当进程起来之后,不断执行jstack <pid>,最终将会看到有非常多的Attach Listener线程

其实问题的根本就是有一个空档期(设置_initialized为true之前)可能存在多次创建线程的可能。

总结

总的来说,创建Attach Listener线程是通过Signal Dispatcher线程来创建的,但是决定Signal Dispatcher是否可以重复创建Attach Listener线程的标记是在某个Attach Listener线程里设置的,如果没有及时设置该标记,就可能存在创建多个Attach Listener线程的情况。

一起来学习吧:

PerfMa KO 系列课之 JVM 参数【Memory篇】

实战:一次疑似内存泄漏的问题排查

作者:PerfMa

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

西园寺花火

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值