死锁是一个很经典的多线程问题,通过下面的代码模拟它出现的场景并记录解决方法。
package org.agoncal.sample.deadlock;
public class DeadLock {
private final Object obj1 = new Object();
private final Object obj2 = new Object();
public void method1() throws InterruptedException
{
synchronized(obj1)
{
Thread.sleep(1000);
synchronized(obj2)
{
System.out.println("method1……");
}
}
}
public void method2() throws InterruptedException
{
synchronized(obj2)
{
Thread.sleep(1000);
synchronized(obj1)
{
System.out.println("method2……");
}
}
}
class Thread1 extends Thread
{
private DeadLock dl;
public Thread1(DeadLock dl)
{
this.dl = dl;
}
@Override
public void run() {
try {
dl.method1();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
class Thread2 extends Thread
{
private DeadLock dl;
public Thread2(DeadLock dl)
{
this.dl = dl;
}
@Override
public void run() {
try {
dl.method2();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
public static void main(String[] args)
{
DeadLock d = new DeadLock();
Thread t1 = new DeadLock().new Thread1(d);
Thread t2 = new DeadLock().new Thread2(d);
t1.start();
t2.start();
}
}
分析方法:
1、找到java进程的pid
2、打印线程快照
介绍一下每一部分的意思,以"Thread-1"为例:
(1)"Thread-1"表示线程名称
(2)"prio=6"表示线程优先级
(3)"tid=0x000000000b05c800"表示线程Id
(4)nid=0x3660
线程对应的本地线程Id,这个重点说明下。因为Java线程是依附于Java虚拟机中的本地线程来运行的,实际上是本地线程在执行Java线程代码,只有本地线程才是真正的线程实体。Java代码中创建一个thread,虚拟机在运行期就会创建一个对应的本地线程,而这个本地线程才是真正的线程实体。Linux环境下可以使用"top -H -p JVM进程Id"来查看JVM进程下的本地线程(也被称作LWP)信息,注意这个本地线程是用十进制表示的,nid是用16进制表示的,转换一下就好了,0x3660对应的本地线程Id应该是13920。
(5)"java.lang.Thread.State:BLOCKED"表示线程的状态
如果出现死锁,jstack命令输出的底部会出现下图的输出: