• 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设计的时候根据不同的操作系

  • 相关阅读:
    css 盒模型
    地级市HVV | 未授权访问合集
    NumPy的实用函数整理之sliding_window_view
    10.SpringBoot实战演练
    牛客小白月赛#59(A~F)
    默认路由,直接路由,静态路由,动态路由
    使用elasticsearch_exporter监控elasticsearch集群的状态
    jenkins 安装 2.377版本主要阻塞问题
    代码随想录Day_55打卡
    2022薪酬调查结果,CRISC和CDPSE更是包揽了冠亚军
  • 原文地址:https://blog.csdn.net/lt_xiaodou/article/details/126813736