网站做电子链接标识申请好吗,网站访问大小 计算流量,wordpress的模板目录在哪里设置,枸杞网站建设方案如何避免Java中的死锁?是流行的Java面试问题之一#xff0c;也是多线程的流行话题之一。尽管问题看起来很简单#xff0c;但是一旦深入#xff0c;大多数Java开发人员就会陷入困境。面试问题以“什么是死锁?”当两个或多个线程正在等待彼此释放所需资源(锁定)并陷入无限时…如何避免Java中的死锁?是流行的Java面试问题之一也是多线程的流行话题之一。尽管问题看起来很简单但是一旦深入大多数Java开发人员就会陷入困境。面试问题以“什么是死锁?”当两个或多个线程正在等待彼此释放所需资源(锁定)并陷入无限时间的阻塞时这种情况称为死锁它只会在多任务或多线程的情况下发生。如何检测Java中的死锁虽然这可能有很多答案但我的版本是如果我看到嵌套的同步块或者从其他同步方法调用一个同步方法或者试图在不同的对象上获得锁那么如果开发人员不是非常小心就很可能出现死锁。另一种方法是在运行应用程序时发现线程被锁定尝试使用线程转储在Linux中可以通过命令“kill -3”来实现这将打印应用程序日志文件中所有线程的状态就可以看到哪个线程被锁定在哪个对象上。也可以使用fastthread等工具分析线程转储上传你的线程转储并分析它。另一种方法是使用jConsole/VisualVM它将准确地显示哪些线程被锁定以及在哪个对象上。编写一个会导致死锁的Java程序一旦回答了前面的问题他们可能会要求你编写导致Java死锁的代码这是我的一个版本/*** Java程序通过强制循环等待来创建死锁** author WINDOWS 8**/public class DeadLockDemo {/** T此方法请求两个锁首先是String然后是Integer*/public void method1() {synchronized (String.class) {System.out.println(Aquired lock onString.classobject);synchronized (Integer.class) {System.out.println(Aquired lock on Integer.class object);}}}/* * 此方法也请求相同的两个锁* 顺序相反先整型然后是字符串* 如果一个线程持有字符串锁则会产生潜在的死锁* 另一个持有整数锁它们会一直等待对方*/public void method2() {synchronized (Integer.class) {System.out.println(Aquired lock on Integer.classobject);synchronized (String.class) {System.out.println(Aquired lock on String.class object);}}}}如果method1()和method2()都将被两个或多个线程调用那么死锁的可能性很大如果线程1在执行method1()时获取Sting对象上的锁而线程2在执行method2()时获取Integer对象上的锁那么线程2将等待对方释放Integer和String上的锁以继续执行而这永远不会发生此图准确演示了我们的程序其中一个线程持有一个对象的锁并等待其他线程持有的其他对象锁。这个图展示了我们的这个程序,一个线程持有一个对象的锁,等待另一个线程持有的其他对象锁。你可以看到线程1想要对线程2持有的对象2的锁定而线程2想要对线程1持有的对象1进行锁定。由于没有线程愿意放弃所以存在死锁并且Java程序卡住了。如何避免Java中的死锁现在面试官来到最后一部分这是我认为最重要的部分之一; 你如何修复代码中的死锁或者如何避免Java中的死锁如果仔细查看了上面的代码那么你可能已经发现死锁的真正原因不是多线程而是它们请求锁定的方式如果提供有序访问则问题将得到解决。这是我的固定版本它避免了由于没有抢占的无效循环等待导致的死锁这是需要死锁的四个条件之一。public class DeadLockFixed {/*** 两种方法现在都以相同的顺序请求锁定首先是Integer然后是String。* 也可以完成反向例如第一个String然后是Integer* 两者都会解决问题只要两种方法都请求锁定* 按照一致的顺序*/public void method1() {synchronized (Integer.class) {System.out.println(Aquired lock on Integer.class object);synchronized (String.class) {System.out.println(Aquired lock on String.class object);}}}public void method2() {synchronized (Integer.class) {System.out.println(Aquired lock on Integer.class object);synchronized (String.class) {System.out.println(Aquired lock on String.class object);}}}}现在不存在任何死锁因为两个方法都以相同的顺序访问Integer和String类文字的锁。因此如果线程A获取对Integer对象的锁定则线程B将不会继续直到线程A释放整数锁定同样即使线程B保持字符串锁定线程A也不会被阻止因为现在线程B不会期望线程A释放整数锁定继续进行。