zoukankan      html  css  js  c++  java
  • jstack排查锁竞争

    排查线程信息的常用工具是jstack,jstack用于生成当前进程的所有线程快照。

    准备

    设计一个程序:两个线程,两个锁,互相等待造成死锁。

    然后使用jstack排查锁竞争情况。

    jstack使用

    一般出现cpu异常的情况下,会使用top命令找到cpu占用较高的进程,得到进程pid。

    如果定位到或怀疑是某个java进程出现锁竞争,可以直接使用jps找到对应的pid,进行一下操作。

    jstack查看线程信息

    jstack -pid

    jstack 18176
    2021-06-08 09:51:39
    Full thread dump Java HotSpot(TM) 64-Bit Server VM (25.31-b07 mixed mode):
    
    "DestroyJavaVM" #14 prio=5 os_prio=0 tid=0x0000000002f36000 nid=0x41f4 waiting on condition [0x0000000000000000]
       java.lang.Thread.State: RUNNABLE
    
    "Thread-1" #13 prio=5 os_prio=0 tid=0x000000001f26f000 nid=0x4724 waiting for monitor entry [0x000000002012f000]
       java.lang.Thread.State: BLOCKED (on object monitor)
            at clc.jvm.Student1.run(AuLock.java:60)
            - waiting to lock <0x000000076b9dd3f0> (a java.lang.String)
            - locked <0x000000076b584598> (a java.lang.Integer)
    
    "Thread-0" #12 prio=5 os_prio=0 tid=0x000000001f1c1000 nid=0x2acc waiting for monitor entry [0x000000002002f000]
       java.lang.Thread.State: BLOCKED (on object monitor)
            at clc.jvm.Student1.run(AuLock.java:50)
            - waiting to lock <0x000000076b584598> (a java.lang.Integer)
            - locked <0x000000076b9dd3f0> (a java.lang.String)
    
    "Service Thread" #11 daemon prio=9 os_prio=0 tid=0x000000001f124000 nid=0x45cc runnable [0x0000000000000000]
       java.lang.Thread.State: RUNNABLE
    
    "C1 CompilerThread3" #10 daemon prio=9 os_prio=2 tid=0x000000001f0a7000 nid=0x46d0 waiting on condition [0x0000000000000000]
       java.lang.Thread.State: RUNNABLE
    
    "C2 CompilerThread2" #9 daemon prio=9 os_prio=2 tid=0x000000001f092000 nid=0x160c waiting on condition [0x0000000000000000]
       java.lang.Thread.State: RUNNABLE
    
    "C2 CompilerThread1" #8 daemon prio=9 os_prio=2 tid=0x000000001ee29000 nid=0x1438 waiting on condition [0x0000000000000000]
       java.lang.Thread.State: RUNNABLE
    
    "C2 CompilerThread0" #7 daemon prio=9 os_prio=2 tid=0x000000001f06c800 nid=0x384c waiting on condition [0x0000000000000000]
       java.lang.Thread.State: RUNNABLE
    
    "Monitor Ctrl-Break" #6 daemon prio=5 os_prio=0 tid=0x000000001ed90000 nid=0x4b3c runnable [0x000000001f92e000]
       java.lang.Thread.State: RUNNABLE
            at java.net.SocketInputStream.socketRead0(Native Method)
            at java.net.SocketInputStream.read(SocketInputStream.java:150)
            at java.net.SocketInputStream.read(SocketInputStream.java:121)
            at sun.nio.cs.StreamDecoder.readBytes(StreamDecoder.java:284)
            at sun.nio.cs.StreamDecoder.implRead(StreamDecoder.java:326)
            at sun.nio.cs.StreamDecoder.read(StreamDecoder.java:178)
            - locked <0x000000076baa2810> (a java.io.InputStreamReader)
            at java.io.InputStreamReader.read(InputStreamReader.java:184)
            at java.io.BufferedReader.fill(BufferedReader.java:161)
            at java.io.BufferedReader.readLine(BufferedReader.java:324)
            - locked <0x000000076baa2810> (a java.io.InputStreamReader)
            at java.io.BufferedReader.readLine(BufferedReader.java:389)
            at com.intellij.rt.execution.application.AppMainV2$1.run(AppMainV2.java:48)
    
    "Attach Listener" #5 daemon prio=5 os_prio=2 tid=0x000000001ea6b800 nid=0x4050 waiting on condition [0x0000000000000000]
       java.lang.Thread.State: RUNNABLE
    
    "Signal Dispatcher" #4 daemon prio=9 os_prio=2 tid=0x000000001eab8800 nid=0x1ed0 runnable [0x0000000000000000]
       java.lang.Thread.State: RUNNABLE
    
    "Finalizer" #3 daemon prio=8 os_prio=1 tid=0x000000000302d000 nid=0x3a48 in Object.wait() [0x000000001ea2e000]
       java.lang.Thread.State: WAITING (on object monitor)
            at java.lang.Object.wait(Native Method)
            - waiting on <0x000000076b5062f8> (a java.lang.ref.ReferenceQueue$Lock)
            at java.lang.ref.ReferenceQueue.remove(ReferenceQueue.java:142)
            - locked <0x000000076b5062f8> (a java.lang.ref.ReferenceQueue$Lock)
            at java.lang.ref.ReferenceQueue.remove(ReferenceQueue.java:158)
            at java.lang.ref.Finalizer$FinalizerThread.run(Finalizer.java:209)
    
    "Reference Handler" #2 daemon prio=10 os_prio=2 tid=0x0000000003026000 nid=0x1d28 in Object.wait() [0x000000001e92f000]
       java.lang.Thread.State: WAITING (on object monitor)
            at java.lang.Object.wait(Native Method)
            - waiting on <0x000000076b505d68> (a java.lang.ref.Reference$Lock)
            at java.lang.Object.wait(Object.java:502)
            at java.lang.ref.Reference$ReferenceHandler.run(Reference.java:157)
            - locked <0x000000076b505d68> (a java.lang.ref.Reference$Lock)
    
    "VM Thread" os_prio=2 tid=0x000000001ca38800 nid=0x424c runnable
    
    "GC task thread#0 (ParallelGC)" os_prio=0 tid=0x0000000002f4c000 nid=0x2720 runnable
    
    "GC task thread#1 (ParallelGC)" os_prio=0 tid=0x0000000002f4d800 nid=0xab0 runnable
    
    "GC task thread#2 (ParallelGC)" os_prio=0 tid=0x0000000002f4f000 nid=0x2c34 runnable
    
    "GC task thread#3 (ParallelGC)" os_prio=0 tid=0x0000000002f50800 nid=0x19b8 runnable
    
    "GC task thread#4 (ParallelGC)" os_prio=0 tid=0x0000000002f53000 nid=0x496c runnable
    
    "GC task thread#5 (ParallelGC)" os_prio=0 tid=0x0000000002f54000 nid=0x1080 runnable
    
    "GC task thread#6 (ParallelGC)" os_prio=0 tid=0x0000000002f58000 nid=0x2ed4 runnable
    
    "GC task thread#7 (ParallelGC)" os_prio=0 tid=0x0000000002f59800 nid=0x47d8 runnable
    
    "VM Periodic Task Thread" os_prio=2 tid=0x000000001f125000 nid=0x1244 waiting on condition
    
    JNI global references: 17
    
    
    Found one Java-level deadlock:
    =============================
    "Thread-1":
      waiting to lock monitor 0x0000000003029f68 (object 0x000000076b9dd3f0, a java.lang.String),
      which is held by "Thread-0"
    "Thread-0":
      waiting to lock monitor 0x000000000302c958 (object 0x000000076b584598, a java.lang.Integer),
      which is held by "Thread-1"
    
    Java stack information for the threads listed above:
    ===================================================
    "Thread-1":
            at clc.jvm.Student1.run(AuLock.java:60)
            - waiting to lock <0x000000076b9dd3f0> (a java.lang.String)
            - locked <0x000000076b584598> (a java.lang.Integer)
    "Thread-0":
            at clc.jvm.Student1.run(AuLock.java:50)
            - waiting to lock <0x000000076b584598> (a java.lang.Integer)
            - locked <0x000000076b9dd3f0> (a java.lang.String)
    
    Found 1 deadlock.
    

    以上为jstack详情,由于程序比较简单,所以线程信息比较短。

    jstack内容分析
    死锁问题排查

    由于程序出现明显的死锁,jstack直接将问题抛出:Found 1 deadlock.

    并且找到出现的线程:

    "Thread-1":

    at clc.jvm.Student1.run(AuLock.java:60)

    - waiting to lock <0x000000076b9dd3f0> (a java.lang.String)

    - locked <0x000000076b584598> (a java.lang.Integer)

    "Thread-0":

    at clc.jvm.Student1.run(AuLock.java:50)

    - waiting to lock <0x000000076b584598> (a java.lang.Integer)

    - locked <0x000000076b9dd3f0> (a java.lang.String)

    <0x000000076b9dd3f0> <>中的十六进制数 表示一个锁的唯一标识,相同的地址表示为同一把锁。

    这段显示:

    1. Thread-1已经锁定了<0x000000076b584598>,但是在等待<0x000000076b9dd3f0>;
    2. Thread-0已经锁定了<0x000000076b9dd3f0>,但是在等待<0x000000076b584598>

    因此这两个线程构成了死锁。

    还能得到信息:两个线程的方法分别是at clc.jvm.Student1.run(AuLock.java:60)和 at clc.jvm.Student1.run(AuLock.java:50);

    这些信息就可以定位程序中死锁的产生位置了。

    其他信息

    查看线程运行状态:
    java.lang.Thread.State: BLOCKED (on object monitor)

    java.lang.Thread.State 会显示此线程的运行状态

    • NEW,未启动的。不会出现在Dump中。
    • RUNNABLE,在虚拟机内执行的。
    • BLOCKED,受阻塞并等待监视器锁。
    • WATING,无限期等待另一个线程执行特定操作。
    • TIMED_WATING,有时限的等待另一个线程的特定操作。
    • TERMINATED,已退出的。

    常见的修饰符信息:

    • locked <地址> 目标:使用synchronized申请对象锁成功,监视器的拥有者。
    • waiting to lock <地址> 目标:使用synchronized申请对象锁未成功,在迚入区等待。
    • waiting on <地址> 目标:使用synchronized申请对象锁成功后,释放锁幵在等待区等待。
    • parking to wait for <地址> 目标
  • 相关阅读:
    nfc相关
    MI卡UID
    VC黑客编程实战视频-破解网吧电影下载
    在WebLogic服务器下 使用Apache Web 服务插件
    C++11FAQ 中文版带完整书签141页PDF
    nginx manager
    JAVA插入数据到MySql少了8小时
    Linux 学习记录
    mssql 用户只能查看授权的数据库
    Mac配置JAVA_HOME
  • 原文地址:https://www.cnblogs.com/chenglc/p/14861854.html
Copyright © 2011-2022 走看看