Java 面试大全〖十〗死锁产生及排查
死锁是指两个或两个以上的进程在执行过程中,因争夺资源而造成的一种互相等待的现象,荐无外力干涉那它们都将无法推进下去,如果系统资源充足,进程的资源请求都能够得到满足,死锁出现的可能性就很低,否则就会因争夺有限的资源而陷入死锁。
产生死锁
package JUC;
import java.util.concurrent.TimeUnit;
class LockDemo implements Runnable{
private String lockA;
private String lockB;
public LockDemo(String lockA, String lockB) {
this.lockA = lockA;
this.lockB = lockB;
}
@Override
public void run() {
synchronized (lockA){
System.out.println(Thread.currentThread().getName()+"\t"+"自己持有: "+lockA+"尝试获得:" +lockB);
try {
TimeUnit.SECONDS.sleep(1); } catch (InterruptedException e) {
e.printStackTrace(); }
synchronized (lockB){
System.out.println(Thread.currentThread().getName()+"\t"+"自己持有: "+lockB+"尝试获得:" +lockA);
}
}
}
}
public class DeadLock {
public static void main(String[] args) {
String lockA="lockA";
String lockB="lockB";
new Thread(new LockDemo(lockA,lockB),"aaa").start();
new Thread(new LockDemo(lockB,lockA),"bbb").start();
}
}
上面产生了一个死锁,下面我们来看怎么分析
- 产生阻塞的原因有很多,你怎么确定是死锁导致的呢?
这里我们借助java自带的一个命令 jps 来查看当前运行的java程序,一步一步排除分析
这里对应着我们的程序里有个pid,拿到pid即可排查
-
1、Jstack命令
jstack是java虚拟机自带的一种堆栈跟踪工具。jstack用于打印出给定的java进程ID或core file或远程调试服务的Java堆栈信息。 Jstack工具可以用于生成java虚拟机当前时刻的线程快照。线程快照是当前java虚拟机内每一条线程正在执行的方法堆栈的集合,生成线程快照的主要目的是定位线程出现长时间停顿的原因,如线程间死锁、死循环、请求外部资源导致的长时间等待等。 线程出现停顿的时候通过jstack来查看各个线程的调用堆栈,就可以知道没有响应的线程到底在后台做什么事情,或者等待什么资源。 -
2、JConsole工具
Jconsole是JDK自带的监控工具,在JDK/bin目录下可以找到。它用于连接正在运行的本地或者远程的JVM,对运行在Java应用程序的资源消耗和性能进行监控,并画出大量的图表,提供强大的可视化界面。而且本身占用的服务器内存很小,甚至可以说几乎不消耗。 ------
使用jstack来打印java运行时异常栈