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

为什么80%的码农都做不了架构师?>>>   hot3.png

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

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

image

2、打印线程快照

多线程死锁问题的模拟及分析_第1张图片

image

介绍一下每一部分的意思,以"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命令输出的底部会出现下图的输出:

多线程死锁问题的模拟及分析_第2张图片

转载于:https://my.oschina.net/zjg23/blog/833391

你可能感兴趣的:(多线程死锁问题的模拟及分析)