Java项目有中多个线程如何查找死锁

作者:沉默着忍受 时间:2023-01-06 06:17:35 

当项目有中多个线程,如何查找死锁?

最近,在IDEA上进行多线程编程中老是在给线程加锁的时候,总是会遇到死锁问题,而当程序出现死锁问题时,编译器不能精确的显示错误的精确位置。当项目代码很多的时候, 往往会给自己添加不必要的麻烦,今天,我就分享分享几个解决方法。

1.编译环境

IDEA 2020 ,windows10, jdk8及以上版本

一、死锁是什么?

死锁指A线程想使用资源但是被B线程占用了,B线程线程想使用资源被A线程占用了,导致程序无法继续下去了。

1.1 死锁的例子;


public class Deadlock {
   public static void main(String[] args) {
       Object lock1 = new Object();
       Object lock2 = new Object();
       Thread thread1 = new Thread(new Runnable() {
           @Override
           public void run() {
               synchronized (lock1){
                   System.out.println("线程一得到了lock1");
                   try{
                       Thread.sleep(1000);
                   } catch (InterruptedException e) {
                       e.printStackTrace();
                   }
                   System.out.println("线程一获取lock2");
                   synchronized (lock2){
                       System.out.println("线程一得到了lock2");
                   }
               }
           }
       });
       thread1.start();

Thread thread2 = new Thread(new Runnable() {
           @Override
           public void run() {
               synchronized (lock2){
                   System.out.println("线程二得到了lock2");
                   try{
                       //让线程2,获取锁1
                       Thread.sleep(1000);
                   } catch (InterruptedException e) {
                       e.printStackTrace();
                   }
                   System.out.println("线程二获取lock1");
                   //尝试获取lock1
                   synchronized (lock1){
                       System.out.println("线程二得到了lock1");
                   }
               }
           }
       });
       thread2.start();
   }
}

Java项目有中多个线程如何查找死锁

1.2 死锁的例子;

形成死锁的条件:
1.互斥条件:(当一个资源被一个线程拥有,当被一个线程拥有后就不能被其他线程所持有)
2.请求拥有条件(一个线程所持有一个资源后又试图请求另一个资源)可修改
3.不可剥夺性:(一个资源被一个线程拥有之后,如果这个线程不释放此资源,那么其他线程不能强制获得此资源)
4.环路等待条件(多个线程在获取资源时形成一个环形链)可修改

二、使用jdk内置工具检测死锁

方法一. jconsole.exe

进入你的jdk安装路径中,打开jdk/bin/jconsole.exe
使用步骤如下:

Java项目有中多个线程如何查找死锁

Java项目有中多个线程如何查找死锁

Java项目有中多个线程如何查找死锁

检测结果:

Java项目有中多个线程如何查找死锁

方法二. jvisualvm.exe

进入你的jdk安装路径中,打开jdk/bin/jvisualvm.exe
优点:比较细,比较全面
缺点:加载有点慢!
使用步骤如下:

Java项目有中多个线程如何查找死锁

Java项目有中多个线程如何查找死锁

可以在里面看到是该项目代码的第39行出现了死锁。

方法三. jmc.exe

进入你的jdk安装路径中,打开jdk/bin/jmc.exe
优点:可以对所以死锁进行判断
缺点:没有给出解决方法
使用步骤如下:

Java项目有中多个线程如何查找死锁

三、死锁解决方法

通过死锁的形成条件来解决死锁问题,从根源上消除死锁。
1.请求拥有条件(一个线程所持有一个资源后又试图请求另一个资源)可修改
2.环路等待条件(多个线程在获取资源时形成一个环形链)可修改

举例修改: 方法:(修改环路等待条件)
//让线程二和线程一竞争同一个锁,修改为并行,这样避免出现环路


public class Deadlock {
   public static void main(String[] args) {
       Object lock1 = new Object();
       Object lock2 = new Object();
       Thread thread1 = new Thread(new Runnable() {
           @Override
           public void run() {
               synchronized (lock1){
                   System.out.println("线程一得到了lock1");
                   try{
                       Thread.sleep(1000);
                   } catch (InterruptedException e) {
                       e.printStackTrace();
                   }
                   System.out.println("线程一获取lock2");
                   synchronized (lock2){
                       System.out.println("线程一得到了lock2");
                   }
               }
           }
       });
       thread1.start();

Thread thread2 = new Thread(new Runnable() {
           @Override
           public void run() {
               synchronized (lock1){   //让线程二和线程一竞争同一个锁,修改为并行,这样避免出现环路
                   System.out.println("线程二得到了lock1");
                   try{
                       //让线程2,获取锁1
                       Thread.sleep(1000);
                   } catch (InterruptedException e) {
                       e.printStackTrace();
                   }
                   System.out.println("线程二获取lock1");
                   //尝试获取lock1
                   synchronized (lock2){
                       System.out.println("线程二得到了lock2");
                   }
               }
           }
       });
       thread2.start();

}
}

Java项目有中多个线程如何查找死锁

四、总结

来源:https://blog.csdn.net/ILOVEMYDEAR/article/details/116027470

标签:Java,多线程,死锁
0
投稿

猜你喜欢

  • 教你使用springSecurity+jwt实现互踢功能

    2023-05-24 19:45:10
  • Spring中实现定时调度的几种方法

    2021-08-29 13:04:44
  • Spring Data JPA 实体类中常用注解说明

    2021-07-21 10:39:38
  • C#使用日志组件log4net

    2022-11-05 03:48:15
  • 图解JVM垃圾内存回收算法

    2023-10-13 17:24:35
  • 如何通过XML方式配置并实现Mybatis

    2021-12-21 02:49:04
  • SpringMVC实现表单验证功能详解

    2023-09-24 09:07:28
  • TableLayout(表格布局)基础知识点详解

    2023-05-30 01:12:42
  • Kotlin Thread线程与UI更新详解

    2021-10-10 00:18:22
  • Java 访问剪切板(复制,粘贴)的示例

    2023-11-10 12:26:13
  • IDEA(jetbrain通用)使用教程图解

    2023-04-15 04:05:49
  • Java实现文件批量重命名,移动和删除

    2022-02-27 03:32:19
  • C#生成唯一值的方法汇总

    2023-11-09 07:25:21
  • C# using的本质及使用详解

    2022-10-10 06:11:23
  • 轻松实现Android语音识别功能

    2022-09-11 17:42:46
  • java实现五子棋程序

    2022-01-29 10:40:38
  • C#根据日期计算星期几的实例代码

    2021-07-24 21:43:34
  • ubuntu环境下反编译android apk的方法

    2022-08-09 13:41:41
  • 简单谈谈android studio 的单元测试

    2021-08-21 14:52:30
  • C#完成word文档打印的方法

    2023-05-29 23:55:36
  • asp之家 软件编程 m.aspxhome.com