您的位置:首页 > 其它

多线程死锁问题的模拟及分析

2017-02-07 00:00 267 查看
死锁是一个很经典的多线程问题,通过下面的代码模拟它出现的场景并记录解决方法。

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命令输出的底部会出现下图的输出:



内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: