线程死锁检测方法

原创 2015年08月12日 20:04:16

多条线程以不同的顺序抢占同步资源的时候,就有可能发生死锁。

如下图所示,线程1持有锁对象A而希望获得锁对象B;另一方面,线程2持有锁对象B而希望

获得锁对象A。并且这两个线程的操作是交错执行的,因此它们会发生死锁。


当发生的死锁后,JDK自带了两个工具(jstack和JConsole),可以用来监测分析死锁的发生原因。

jstack工具用于生于生成虚拟机当前时刻的线程快照。线程快照就是当前虚拟机每一条线程正在

执行的方法堆栈的集合,生成快照可以用于定位诸如线程死锁、死循环等问题。

JConsole是一种可视化监视管理工具。用于连接正在运行的JVM进程,以监控 Java 应用程序性

能和跟踪 Java 中的代码。

下面以一个死锁例子来说明如何使用这两个工具来分析线程死锁。

死锁示例代码如下:

class SynThread implements Runnable{
	
	int a ,b;
	
	public SynThread(int a,int b){
		this.a=a;
		this.b=b;
	}
	@Override
	public void run() {
		synchronized (Integer.valueOf(a)) {//必须用valueOf()方法
			synchronized (Integer.valueOf(b)) {
				System.err.println("a+b=="+(a+b));
			}
		}
	}
	
}

public class entry {

	public static void main(String[] args) {
		
		//循环主要是为了加大死锁概率
		for(int i=0;i<100;i++){
			new Thread(new SynThread(1,2)).start();
			new Thread(new SynThread(2,1)).start();
		}
	}
}

说明:以上代码有可能发生死锁,原因是Integer.valueOf()方法作了缓存优化,对[-128,127]之间

的数字会被缓存。也就是说,循环代码中一共只创建了两个不同的对象。假设在两个synchronized

块之间发生了线程切换,那就有可能造成,线程A等待被线程B持有Integer.valueOf(1)对象,线程B

等待被线程A持有Integer.valueOf(2)对象,结果出现了死锁。(可能需要多次执行直到程序出现

阻塞现象)


JDK源代码Integer.java类对valueOf()方法的优化细节如下

    public static Integer valueOf(int i) {
        if(i >= -128 && i <= IntegerCache.high)
            return IntegerCache.cache[i + 128];
        else
            return new Integer(i);
    }

  • 使用jstack工具导出线程堆栈信息(以Windows环境为例)

1.找到运行当前程序的JVM的进程id,命令及结果如下



2.运行jstack命令,并将结果信息导出来


3.使用文本编辑器打开刚导出的文本,只要查看最后关于死锁的堆栈信息即可


4.直接从堆栈信息不能直观得到结论,没关系,我们可以画图理清线程间的调度情况

出现闭合环路,发生死锁


  • 使用JConsole可视化工具检测死锁

1.直接执行JConsole工具(在jdk/bin目录下),选择目标JVM进程,然后点击连接


2.切换到“线程”标签页,点击“检测死锁”按钮



3.选择不同的线程,可以查看其资源调度信息,采用类似的方法分析,可以得出与

采用jstack分析一样的结论




版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

死锁的检测和解除

死锁的检测和解除

死锁怎么检测?

先OO分析、建模,再研究具体问题。这里面涉及的对象: 1)线程 2)锁 关系: 占有:线程——锁 的一对多关系:一个线程可以占有多个锁,一个锁只能被一个线程占有 等待:线程——锁 的多对多关系:一个线...

构成死锁的必要条件是什么?如何检测死锁,解除死锁?

产生死锁的原因主要是: (1) 因为系统资源不足。 (2) 进程运行推进的顺序不合适。 (3) 资源分配不当等。 如果系统资源充足,进程的资源请求都能够得到满足,死锁出现的可能性就很低,否则 ...

什么是死锁以及死锁的预防、检测与修复

来自http://blog.163.com/yanenshun@126/blog/static/128388169200982444858590/?fromdm&fromSearch&isFromSe...

多线程 —— 死锁代码展示

死锁的意思其实就是相互等待。一个生活中的简单例子:我们去包子铺吃包子。客户坚持:先吃包子,后付钱。 卖家坚持:先付钱以后,才能吃包子。两边如果都坚持自己的原则,对于客户买包子,卖家卖包子赚钱这件事都...

多线程编程(一)——写一个简单的死锁

(整个九月忙着找工作,好多收获,好多遗憾,最终结局还可以接受,技术路还很远,再接再厉!面去哪儿网时,写惯了算法的我突然让写了几个多线程编程,有点蒙蔽,最近好好整理一下) 死锁发生的原因: 1、系统资源...

死锁原理及代码

死锁是两个或更多线程阻塞着等待其它处于死锁状态的线程所持有的锁。死锁通常发生在多个线程同时但以不同的顺序请求同一组锁的时候。 例如,如果线程1锁住了A,然后尝试对B进行加锁,同时线程2已...

多线程死锁例子

一.例子  两个资源对象o1,o2,两个线程t1,t2。 设置t1的初始条件为真,线程启动后。t1获取o1锁,然后等待o2。然后获取o2锁,然后释放o2锁。接着释放o1锁。程序结束 设置t2的初...

多线程死锁及解决办法

死锁是由于不同线程按照不同顺序进行加锁而造成的。如: 线程A:对lock a加锁 => 对lock b加锁 => dosth => 释放lock b => 释放lock a 线程B:对...
  • zgaoq
  • zgaoq
  • 2017-05-18 17:22
  • 506

如何检测死锁并快速定位死锁位置

在游戏中有时会遇到这样一种情况,某客户端发了个请求到服务端,但收不到服务端回复,看服务端的log,也没任何错误,最后调试跟踪代码,发现代码死锁了。遇到这种情况比较纠结,于是捣腾了一个自动检测死锁的功能...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)