• JVM源码解析Java Attach处理流程


    前言

    当Java程序运行时出现CPU负载高、内存占用大等异常情况时,通常需要使用JDK自带的工具jstack、jmap查看JVM的运行时数据,并进行分析。

    什么是Java Attach

    那么JVM自带的这些工具是如何获取到JVM的相关信息呢?
    JVM提供了 Java Attach 功能,能够让客户端与目标JVM进行通讯从而获取JVM运行时的数据,甚至可以通过Java Attach 加载自定义的代理工具,实现AOP、运行时class热更新等功能。

    如果我们通过jstack打印线程栈的时候会发现有这么2个线程:Signal Dispatcher和Attach Listener。

    1. "Signal Dispatcher" #4 daemon prio=9 os_prio=2 cpu=0.00ms elapsed=917.19s tid=0x00000164ff377000 nid=0x4ba0 runnable [0x0000000000000000]
    2. java.lang.Thread.State: RUNNABLE
    3. "Attach Listener" #5 daemon prio=5 os_prio=2 cpu=0.00ms elapsed=917.19s tid=0x000001648f4d1800 nid=0x1fc0 waiting on condition [0x0000000000000000]
    4. java.lang.Thread.State: RUNNABLE

    Signal Dispatcher用于处理操作系统信号(软中断信号),Attach Listener线程用于JVM进程间的通信。

    操作系统支持的信号可以通过kill -l查看。比如我们平时杀进程用kill -9 可以看到9对应的信号就是SIGKILL。
    其他的信号并不会杀掉JVM进程,而是通知到进程, 具体进程如何处理根据Signal Dispatcher线程处理逻辑决定。

    1. root@DESKTOP-45K54QO:~# kill -l
    2. 1) SIGHUP 2) SIGINT 3) SIGQUIT 4) SIGILL 5) SIGTRAP
    3. 6) SIGABRT 7) SIGBUS 8) SIGFPE 9) SIGKILL 10) SIGUSR1
    4. 11) SIGSEGV 12) SIGUSR2 13) SIGPIPE 14) SIGALRM 15) SIGTERM
    5. 16) SIGSTKFLT 17) SIGCHLD 18) SIGCONT 19) SIGSTOP 20) SIGTSTP
    6. 21) SIGTTIN 22) SIGTTOU 23) SIGURG 24) SIGXCPU 25) SIGXFSZ
    7. 26) SIGVTALRM 27) SIGPROF 28) SIGWINCH 29) SIGIO 30) SIGPWR
    8. 31) SIGSYS 34) SIGRTMIN 35) SIGRTMIN+1 36) SIGRTMIN+2 37) SIGRTMIN+3
    9. 38) SIGRTMIN+4 39) SIGRTMIN+5 40) SIGRTMIN+6 41) SIGRTMIN+7 42) SIGRTMIN+8
    10. 43) SIGRTMIN+9 44) SIGRTMIN+10 45) SIGRTMIN+11 46) SIGRTMIN+12 47) SIGRTMIN+13
    11. 48) SIGRTMIN+14 49) SIGRTMIN+15 50) SIGRTMAX-14 51) SIGRTMAX-13 52) SIGRTMAX-12
    12. 53) SIGRTMAX-11 54) SIGRTMAX-10 55) SIGRTMAX-9 56) SIGRTMAX-8 57) SIGRTMAX-7
    13. 58) SIGRTMAX-6 59) SIGRTMAX-5 60) SIGRTMAX-4 61) SIGRTMAX-3 62) SIGRTMAX-2
    14. 63) SIGRTMAX-1 64) SIGRTMAX

    线程初始化

    在虚拟机初始完成后,Signal Dispatcher和Attach Listener线程会根据配置进行必要的初始化。

    1. jint Threads::create_vm(JavaVMInitArgs* args, bool* canTryAgain) {
    2. ...
    3. //记录虚拟机初始化完成时间
    4. Management::record_vm_init_completed();
    5. ...
    6. // 初始化Signal Dispatcher
    7. os::signal_init();
    8. // 当设置了StartAttachListener或者无法懒加载时启动Attach Listener
    9. if (!DisableAttachMechanism) {
    10. AttachListener::vm_start();
    11. if (StartAttachListener || AttachListener::init_at_startup()) {
    12. AttachListener::init();
    13. }
    14. }
    15. ...
    16. // 通知所有的 JVMTI agents 虚拟机初始化完成
    17. JvmtiExport::post_vm_initialized();
    18. ...
    19. }

    相关JVM参数

    JVM相关参数如下,默认都是false

    JVM参数

    默认值

    DisableAttachMechanism

    false

    StartAttachListener

    false

    ReduceSignalUsage

    false

    除了这三个参数以外,我们可以看到
    AttachListener::init_at_startup()也是用于控制Attach Listener是否初始化。
    JDK设计的时候根据不同的操作系

  • 相关阅读:
    【数据库】时区及JDBC的时区设置
    【测试】自动化UI测试小工具
    Xcode项目文件与资源管理:精通技巧与实践指南
    5G/4G物联网无线通信模块
    MFC项目改为多字节字符集界面风格变为win98风格的问题
    pivot转换表
    深度学习入门(二十八)卷积神经网络——AlexNet
    LDA算法并提取这份数据集中各个文档的主题
    面试突击78:@Autowired 和 @Resource 有什么区别?
    ABP框架之——数据访问基础架构(下)
  • 原文地址:https://blog.csdn.net/lt_xiaodou/article/details/126813736