我的编程空间,编程开发者的网络收藏夹
学习永远不晚

如何通过编程发现Java死锁

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

如何通过编程发现Java死锁

今天就跟大家聊聊有关如何通过编程发现Java死锁,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。

死锁是指,两个或多个动作一直在等待其他动作完成而使得所有动作都始终处在阻塞的状态。想要在开发阶段检测到死锁是非常困难的,而想要解除死锁往往需要重新启动程序。更糟的是,死锁通常发生在负载最重的生产过程中,而想要在测试中发现它,十分不易。之所以这么说,是因为测试线程之间所有可能的交叉是不现实的。尽管出现了一些静态分析库可以帮助我们发现可能出现的死锁,我们还是有必要在运行时检测到死锁,并且得到有用的信息,以便我们解决这个问题或者重启程序,或者做些其他的事情。

在编程中使用ThreadMXBean类来检测死锁

Java  5引入了ThreadMXBean接口,它提供了多种监视线程的方法。我建议您了解所有这些方法,因为当您没使用外部工具时,它们会为您提供很多有用的操作以便您监测程序性能。这里,我们感兴趣的方法是findMonitorDeadlockedThreads,如过您使用的是Java  6,对应的方法是findDeadlockedThreads。二者的区别的是,findDeadlockedThreads还可以检测到owner  locks(java.util.concurrent)引起的死锁,而findMonitorDeadlockedThreads只能检测monitor  locks(例如,同步块)。由于保留老版本的方法只是出于兼容性的考虑,所以我将使用新版本的方法。在这里,编程的思想是把对死锁的周期性检测封装到一个可重用组件里,之后我们只需启动它、随它去。

一种实现调度的方法是通过执行器框架,即一组良好抽象并易于使用的多线程类。

ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);  this.scheduler.scheduleAtFixedRate(deadlockCheck, period, period, unit);

就是那么简单,在我们通过选择周期和时间单位而设置了一个特定时间后,就得到了一个周期性调用的线程。接着,我们想使功用得以拓展从而允许用户提供在程序检测到死锁时所触发的行为。***,我们需要一个方法来接收用于描述死锁中所有线程的一系列对象。

void handleDeadlock(final ThreadInfo deadlockedThreads);

现在,实现死锁检测类已经万事俱备了。

public interface DeadlockHandler { void handleDeadlock(final ThreadInfo  deadlockedThreads); } public class DeadlockDetector { private final  DeadlockHandler deadlockHandler; private final long period; private final  TimeUnit unit; private final ThreadMXBean mbean =  ManagementFactory.getThreadMXBean; private final ScheduledExecutorService  scheduler = Executors.newScheduledThreadPool(1); final Runnable deadlockCheck =  new Runnable { @Override public void run { long deadlockedThreadIds =  DeadlockDetector.this.mbean.findDeadlockedThreads; if (deadlockedThreadIds !=  null) { ThreadInfo threadInfos =  DeadlockDetector.this.mbean.getThreadInfo(deadlockedThreadIds);  DeadlockDetector.this.deadlockHandler.handleDeadlock(threadInfos); } } }; public  DeadlockDetector(final DeadlockHandler deadlockHandler, final long period, final  TimeUnit unit) { this.deadlockHandler = deadlockHandler; this.period = period;  this.unit = unit; } public void start { this.scheduler.scheduleAtFixedRate(  this.deadlockCheck, this.period, this.period, this.unit); } }

让我们动手试试。首先,我们要创建一个handler用来向System.err输出死锁线程的信息。在现实场景中,我们可以用它发送邮件,比如:

public class DeadlockConsoleHandler implements DeadlockHandler { @Override  public void handleDeadlock(final ThreadInfo deadlockedThreads) { if  (deadlockedThreads != null) { System.err.println("Deadlock detected!"); Map  stackTraceMap = Thread.getAllStackTraces; for (ThreadInfo threadInfo :  deadlockedThreads) { if (threadInfo != null) { for (Thread thread :  Thread.getAllStackTraces.keySet) { if (thread.getId == threadInfo.getThreadId) {  System.err.println(threadInfo.toString.trim); for (StackTraceElement ste :  thread.getStackTrace) { System.err.println("t" + ste.toString.trim); } } } } } }  } }

这一过程在所有的堆栈追踪中反复进行并为每个线程信息打印对应的堆栈踪迹。通过这种方式,我们可以准确知道每个线程等待的位置和对象。但这个方法有一个缺陷——当一个线程只是暂时等待时,可能会被当作一个暂时的死锁,从而引发错误的警报。出于此,当我们处理死锁时,原始线程不能继续存在而findDeadlockedThreads方法会返回没有此类线程。为了避免可能出现的NullPointerException,我们需要警惕这种情况。***,让我们促成一个死锁来看看系统是如何运行的。

DeadlockDetector deadlockDetector = new DeadlockDetector(new  DeadlockConsoleHandler, 5, TimeUnit.SECONDS); deadlockDetector.start; final  Object lock1 = new Object; final Object lock2 = new Object; Thread thread1 = new  Thread(new Runnable { @Override public void run {synchronized (lock1) {  System.out.println("Thread1 acquired lock1"); try {  TimeUnit.MILLISECONDS.sleep(500); } catch (InterruptedException ignore) { }  synchronized (lock2) { System.out.println("Thread1 acquired lock2"); } } } });  thread1.start; Thread thread2 = new Thread(new Runnable { @Override public void  run { synchronized (lock2) { System.out.println("Thread2 acquired lock2");  synchronized (lock1) { System.out.println("Thread2 acquired lock1"); } } } });  thread2.start;

输出:

Thread1 acquired lock1 Thread2 acquired lock2 Deadlock detected! “Thread-1”  Id=11 BLOCKED on java.lang.Object@68ab95e6 owned by “Thread-0” Id=10  deadlock.DeadlockTester$2.run(DeadlockTester.java:42)  java.lang.Thread.run(Thread.java:662) “Thread-0” Id=10 BLOCKED on  java.lang.Object@58fe64b9 owned by “Thread-1” Id=11  deadlock.DeadlockTester$1.run(DeadlockTester.java:28)  java.lang.Thread.run(Thread.java:662)

记住,死锁检测的开销可能会很大,你需要用你的程序来测试一下你是否真的需要死锁检测以及多久检测一次。我建议死锁检测的时间间隔至少为几分钟,因为更加频繁的检测并没有太大的意义,原因是我们并没有一个复原计划,我们能做的只是调试和处理错误或者重启程序并祈祷不会再次发生死锁。如果你有关于解决死锁问题的好建议或者关于这个解决方案的疑问,请在下面留言。

看完上述内容,你们对如何通过编程发现Java死锁有进一步的了解吗?如果还想了解更多知识或者相关内容,请关注编程网行业资讯频道,感谢大家的支持。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

如何通过编程发现Java死锁

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

如何通过编程发现Java死锁

今天就跟大家聊聊有关如何通过编程发现Java死锁,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。死锁是指,两个或多个动作一直在等待其他动作完成而使得所有动作都始终处在阻塞的状态。想要在
2023-06-17

Java并发编程中死锁的实现

这篇文章给大家介绍Java并发编程中死锁的实现,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。一、什么是死锁所谓死锁是指多个线程因竞争资源而造成的一种僵局(互相等待),若无外力作用,这些进程都将无法向前推进二、死锁产生的
2023-06-15

java并发编程死锁定义及避免死锁案例分析

这篇文章主要介绍“java并发编程死锁定义及避免死锁案例分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“java并发编程死锁定义及避免死锁案例分析”文章能帮助大家解决问题。场景模拟分析场景一:狭路
2023-06-29

Java 内存模型与死锁:深入理解并发编程中的死锁问题

本文深入探讨 Java 内存模型与死锁问题之间的关联,并以示例代码阐释死锁的成因和解决方法,旨在帮助读者深入理解并发编程中的死锁问题。
Java 内存模型与死锁:深入理解并发编程中的死锁问题
2024-02-04

Java编程之多线程死锁与线程间通信简单实现代码

死锁定义 死锁是指两个或者多个线程被永久阻塞的一种局面,产生的前提是要有两个或两个以上的线程,并且来操作两个或者多个以上的共同资源;我的理解是用两个线程来举例,现有线程A和B同时操作两个共同资源a和b,A操作a的时候上锁LockA,继续执行
2023-05-30

C#开发中如何处理并发编程和死锁问题

C#开发中如何处理并发编程和死锁问题,需要具体代码示例摘要:并发编程是现代软件开发中的重要主题,但也带来了一些挑战,例如线程安全、竞态条件和死锁等问题。本文将重点讨论在C#开发中处理并发编程和死锁问题的一些方法,并给出具体的代码示例。引言:
2023-10-22

Java多线程环境下如何实现死锁

这篇文章主要介绍Java多线程环境下如何实现死锁,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!1、死锁产生的条件互斥:一次只有一个进程可以使用一个资源。其他进程不能访问已分配给其他进程的资源。不可抢占:不能抢占进程已
2023-06-25

java多线程死锁如何解决

Java中死锁的解决办法有以下几种:1. 避免使用多个锁:当多个线程需要获取多个锁时,可以尝试将多个锁合并为一个锁,或者将一个锁拆分为多个锁,以避免死锁的发生。2. 保持锁的顺序一致:当多个线程需要获取多个锁时,确保它们获取锁的顺序是一致的
2023-08-24

C++并发编程:如何识别和解决死锁问题?

在 c++++ 并发编程中,死锁问题发生在一或多个线程无限期等待其他线程释放资源时,导致程序挂起。我们可以使用 std::lock_guard 和 std::unique_lock 实现死锁检测,如果发生死锁,会抛出 std::system
C++并发编程:如何识别和解决死锁问题?
2024-05-04

C++ 函数如何解决并发编程中的死锁问题?

在 c++++ 中,使用互斥量函数可以解决多线程并发编程中的死锁问题。具体步骤如下:创建一个互斥量;当线程需要访问共享变量时,获得互斥量;修改共享变量;释放互斥量。这样可以确保任何时刻只有一个线程访问共享变量,有效防止死锁。利用 C++ 函
C++ 函数如何解决并发编程中的死锁问题?
2024-04-26

Go并发编程中如何避免死锁和竞态条件

在 go 并发编程中,为了避免死锁和竞态条件,有以下准则:避免死锁:识别共享资源,指派明确所有者,并使用死锁检测工具。避免竞态条件:使用互斥锁、读写锁或原子操作来确保共享数据的并发访问安全。Go并发编程中避免死锁和竞态条件并发编程涉及多个
Go并发编程中如何避免死锁和竞态条件
2024-05-13

如何通过sysprocesses 简单查询死锁及解决死锁最新方案(四步解决)

如何通过sysprocesses查询和解决死锁(四步方案)识别死锁:查询死锁进程,包括进程ID、等待类型和状态。获取详细信息:获取死锁进程的输入缓冲区,包含导致死锁的资源。终止死锁:根据需要终止死锁进程。防止死锁:实施措施,例如使用事务、优化索引和调整隔离级别,以减少死锁发生的可能性。
如何通过sysprocesses 简单查询死锁及解决死锁最新方案(四步解决)
2024-04-02

Java项目中出现死锁如何解决

Java项目中出现死锁如何解决?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。死锁是这样一种情形:多个线程同时被阻塞,它们中的一个或者全部都在等待某个资源被释放。由于线程被无
2023-05-31

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录