Java线程死锁的问题解决

 更新时间:2026年03月19日 09:28:30   作者:ZePingPingZe  
本文主要介绍了Java线程死锁的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

1. 死锁案例

死锁:线程之间互相持有对方的锁,并且等待获取对方持有的锁。

案例如下:我这里直接写的service层代码,没写测试类(测试类直接调用deadLockTest()方法)。

package cn.sh.tools.service.component.lock;
 
import lombok.AllArgsConstructor;
import lombok.Getter;
import lombok.extern.slf4j.Slf4j;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;
import org.springframework.stereotype.Component;
 
import javax.annotation.Resource;
import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.concurrent.CountDownLatch;
 
/**
 * @Description
 * @Classname LockComponent
 * @Date 2023/4/28
 * @Author sh
 **/
@Component
@Slf4j
public class LockComponent {
 
    // 记得在 yml文件中配置线程池相关参数(核心线程数、工作队列大小、最大线程数)。
    @Resource(name = "applicationTaskExecutor")
    private ThreadPoolTaskExecutor threadPoolTaskExecutor;
 
    // 入参createThreadNum:要创建的线程数。尽量设置个十几二十个,设置的太小可能程序执行太快就发生不了死锁。
    public void deadlockTest(int createThreadNum) {
        // 计数器
        CountDownLatch sunCdl = new CountDownLatch(createThreadNum);
        // 1- 填充数据
        List<DeadLockObj> deadLockObjList = new ArrayList<>(createThreadNum);
        /**
         * for循环 创建DeadLockObj对象,这里只创建了两种对象。
         * 这里创建两种对象的原因是为了 后面创建的线程互相持有锁。
         */
        for (int i = 0; i < createThreadNum; i++) {
            if (i % 2 != 0) {
                // i 为 奇数时,给 DeadLockObj 对象的 a = 1,b = 2。
                deadLockObjList.add(new DeadLockObj(i, 1, 2));
            } else {
                // i 为 偶数时,给 DeadLockObj 对象的 a = 2,b = 1。
                deadLockObjList.add(new DeadLockObj(i, 2, 1));
            }
        }
        deadLockObjList.sort(Comparator.comparingInt(DeadLockObj::getId));
 
        // 循环数据,调用 deadLockInternal()方法,展示死锁情况。
        deadLockObjList.forEach(deadLockObj -> {
            this.threadPoolTaskExecutor.execute(() -> {
                try {
                    sunCdl.await();
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
 
                // 在高并发情况下,必然会出现死锁。
                this.deadLockInternal(deadLockObj);
            });
        });
 
        // 循环将 计数器 减1,当减为0时,上面的线程就开始 真正的并发执行。
        for (int i = 0; i < createThreadNum; i++) {
            sunCdl.countDown();
        }
    }
 
    private void deadLockInternal(DeadLockObj deadLockObj) {
        /**
         * ① 线程A 持有 Integer.valueOf(1),线程B 持有 Integer.valueOf(2)
         * ② 线程A 抢占 Integer.valueOf(2),但是2被线程B持有。
         *   同时,线程B 抢占 Integer.valueOf(1),但是1被线程A持有。
         * 线程A 和 线程B 要抢占的锁都被对方持有,所以,必然会出现死锁。
         */
 
        // 第一把锁
        synchronized (Integer.valueOf(deadLockObj.getA())) {
            // 第二把锁
            synchronized (Integer.valueOf(deadLockObj.getB())) {
                System.out.println(deadLockObj.getId());
            }
        }
    }
 
    @Getter
    @AllArgsConstructor
    static class DeadLockObj {
        Integer id;
        int a;
        int b;
    }
 
}

2. 死锁线程查看

2.1 第一步

当 java程序发生死锁时,JavaVisualVM 的 "线程"标签页 会自动显示红色提示信息:“检测到死锁!生成一个线程dump以获取更多信息”。如下图所示:

JavaVisualVM 不能很方便的查看到到底是哪几个线程发生了死锁,所以,接下来我们就可以用到 JConsole了。

【备注】:JavaVisualVM 在JDK8的时候还是自带工具,到JDK11的时候就不是自带的了,但还是可以免费下载和免费试用的。(至于JDK9、10有没有自带不知道,没下载过)。

2.2 第二步

上面一步,JavaVisualVM 已经【自动】给我们提示出现了【死锁】。这里,我们就可以用 JConsole 来查看了。

死锁 小标签页,我们随便点击一个线程就可以查看对应的 线程快照信息。

到此这篇关于Java线程死锁的问题解决的文章就介绍到这了,更多相关Java线程死锁内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • SpringBoot的application.yml不生效问题及解决

    SpringBoot的application.yml不生效问题及解决

    这篇文章主要介绍了SpringBoot的application.yml不生效问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • Java实现获取小程序带参二维码并保存到本地

    Java实现获取小程序带参二维码并保存到本地

    这篇文章主要介绍了Java实现获取小程序带参二维码并保存到本地,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10
  • Java吃货联盟订餐系统代码实例

    Java吃货联盟订餐系统代码实例

    这篇文章主要介绍了Java订餐系统,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-04-04
  • Java数据结构与算法学习之循环链表

    Java数据结构与算法学习之循环链表

    循环链表是另一种形式的链式存储结构。它的特点是表中最后一个结点的指针域指向头结点,整个链表形成一个环。本文将为大家详细介绍一下循环链表的特点与使用,需要的可以了解一下
    2021-12-12
  • Spring Boot 2.7.8 原生 SSE 服务端开发实战指南

    Spring Boot 2.7.8 原生 SSE 服务端开发实战指南

    在Spring Boot 2.7.8版本中,我们可以利用其原生支持,轻松实现SSE服务,从而为应用提供实时数据推送能力,本文给大家介绍Spring Boot 2.7.8 原生 SSE 服务端开发实战指南,感兴趣的朋友一起看看吧
    2026-04-04
  • maven报错:Failed to execute goal on project问题及解决

    maven报错:Failed to execute goal on p

    这篇文章主要介绍了maven报错:Failed to execute goal on project问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-04-04
  • Java多线程读写锁ReentrantReadWriteLock类详解

    Java多线程读写锁ReentrantReadWriteLock类详解

    本文详细讲解了Java多线程读写锁ReentrantReadWriteLock类,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-12-12
  • 从基础实现到性能优化详解Java实现Word转HTML的完整指南

    从基础实现到性能优化详解Java实现Word转HTML的完整指南

    本文将基于实际项目经验,介绍 Java 中 Word 转 HTML 的几种实现方案,并以其中一种为例详细讲解转换过程中的关键配置,性能优化及常见问题处理
    2026-04-04
  • JavaWeb实现简单查询商品功能

    JavaWeb实现简单查询商品功能

    这篇文章主要为大家详细介绍了JavaWeb实现简单查询商品功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07
  • Servlet实现点击计数器的方法

    Servlet实现点击计数器的方法

    这篇文章主要介绍了Servlet实现点击计数器的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-08-08

最新评论