how2j.cn

当业务比较复杂,多线程应用里有可能会发生死锁

本视频是解读性视频,所以希望您已经看过了本知识点的内容,并且编写了相应的代码之后,带着疑问来观看,这样收获才多。 不建议一开始就观看视频



3分45秒
本视频采用html5方式播放,如无法正常播放,请将浏览器升级至最新版本,推荐火狐,chrome,360浏览器 如果装有迅雷,播放视频呈现直接下载状态,请调整 迅雷系统设置-基本设置-启动-监视全部浏览器 (去掉这个选项)



步骤 1 : 演示死锁   
步骤 2 : 练习-死锁   
步骤 3 : 答案-死锁   

步骤 1 :

演示死锁

1. 线程1 首先占有对象1,接着试图占有对象2
2. 线程2 首先占有对象2,接着试图占有对象1
3. 线程1 等待线程2释放对象2
4. 与此同时,线程2等待线程1释放对象1
就会。。。一直等待下去,直到天荒地老,海枯石烂,山无棱 ,天地合。。。
演示死锁
package multiplethread; import charactor.Hero; public class TestThread { public static void main(String[] args) { final Hero ahri = new Hero(); ahri.name = "九尾妖狐"; final Hero annie = new Hero(); annie.name = "安妮"; Thread t1 = new Thread(){ public void run(){ //占有九尾妖狐 synchronized (ahri) { System.out.println("t1 已占有九尾妖狐"); try { //停顿1000毫秒,另一个线程有足够的时间占有安妮 Thread.sleep(1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } System.out.println("t1 试图占有安妮"); System.out.println("t1 等待中 。。。。"); synchronized (annie) { System.out.println("do something"); } } } }; t1.start(); Thread t2 = new Thread(){ public void run(){ //占有安妮 synchronized (annie) { System.out.println("t2 已占有安妮"); try { //停顿1000秒,另一个线程有足够的时间占有暂用九尾妖狐 Thread.sleep(1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } System.out.println("t2 试图占有九尾妖狐"); System.out.println("t2 等待中 。。。。"); synchronized (ahri) { System.out.println("do something"); } } } }; t2.start(); } }
步骤 2 :

练习-死锁

Or  姿势不对,事倍功半! 点击查看做练习的正确姿势
3个同步对象a, b, c
3个线程 t1,t2,t3

故意设计场景,使这3个线程彼此死锁
步骤 3 :

答案-死锁

在查看答案前,尽量先自己完成,碰到问题再来查看答案,收获会更多
本视频是解读性视频,所以希望您已经看过了本答案的内容,带着疑问来观看,这样收获才多。 不建议一开始就观看视频

3分18秒 本视频采用html5方式播放,如无法正常播放,请将浏览器升级至最新版本,推荐火狐,chrome,360浏览器 如果装有迅雷,播放视频呈现直接下载状态,请调整 迅雷系统设置-基本设置-启动-监视全部浏览器 (去掉这个选项)


package multiplethread; public class TestThread { public static void main(String[] args) { Object a = new Object(); Object b = new Object(); Object c = new Object(); Thread t1 =new Thread(){ public void run(){ synchronized (a) { try { Thread.sleep(1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } synchronized (b) { synchronized (c) { } } } } }; Thread t2 =new Thread(){ public void run(){ synchronized (c) { try { Thread.sleep(1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } synchronized (a) { synchronized (b) { } } } } }; Thread t3 =new Thread(){ public void run(){ synchronized (b) { try { Thread.sleep(1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } synchronized (c) { synchronized (a) { } } } } }; t1.start(); t2.start(); t3.start(); } }


HOW2J公众号,关注后实时获知布最新的教程和优惠活动,谢谢。


问答区域    
2017-11-03 时间啊
西瓜胖子
sleep(1000);应该是一秒吧




1 个答案

cangogo 答案时间:2017-11-23
1000的单位为毫秒,正确。




答案 或者 代码至少填写一项, 如果是自己有问题,请重新提问,否则站长有可能看不到




2016-07-10 代码没实现死锁,不知道问题在哪
小白



输出结果没体现出死锁。问题出现在哪
package Thread.Lock;

public class DeadLock  {
	public static void main(String[] args) {
		TestDeadLock dl1 = new TestDeadLock(true);
		TestDeadLock dl2 = new TestDeadLock(false);
		dl1.start();
		dl2.start();
	}

}


package Thread.Lock;

public class TestDeadLock extends Thread{
	 Object lock1 = new Object();
	 Object lock2 = new Object();
	public boolean flag ;
	public TestDeadLock(boolean flag){
		this.flag=flag;
	}	
	public void run(){
		if(flag){
			synchronized(lock1){
				System.out.println("锁定ob1");
			
			synchronized(lock2){
				System.out.println("锁定ob2");
			}
			}
		}else{
			synchronized(lock2){
				System.out.println("锁定ob2");
			
			synchronized(lock1){
				System.out.println("锁定ob1");
			}
			}
		}
	}
}
输出:
锁定ob1
锁定ob2
锁定ob2
锁定ob1


3 个答案

tccccc 答案时间:2017-10-30
代码看着有点难受,楼下说的对,不妨试试写个==输出,看两个对象的LOCK1是不是同一个对象

JVNewer 答案时间:2017-10-22
不是运行太快。。。你的两个线程分别创建了lock1和lock2,这一共是四个对象,所以你的两个线程的同步对象并不同。当然,线程运行太快可能也不会发生死锁,所以加入sleep更容易发生死锁

thomas 答案时间:2016-07-12
因为代码运行得太快了,第二个线程还没有来得及占有lock2, 第一个线程就已经 完成了占有lock1和lock2的步骤,并且按照顺序,已经释放了lock2和lock1.

这个时候, 第二个线程再来占有lock2的时候,发行并没有任何线程占用。

参考演示代码的第19行,给一点sleep时间,让第二个线程有充足的时间占用lock2




答案 或者 代码至少填写一项, 如果是自己有问题,请重新提问,否则站长有可能看不到









提问之前请登陆
关于 JAVA 中级-多线程-死锁 的提问

尽量提供截图代码异常信息,有助于分析和解决问题。 也可进本站QQ群交流: 389538688
站长会在每个工作日早上尽量回答提问(如果有漏掉没有回答的,请进群提醒一下)
提问尽量提供完整的代码,环境描述,越是有利于问题的重现,您的问题越能更快得到解答。
对教程中代码有疑问,请提供是哪个步骤,哪一行有疑问,这样便于快速定位问题,提高问题得到解答的速度
站长是玻璃心,提问的时候请语气温柔些 kiss~
截止2017-5-19日累计提问 1638个,站长回答了 1546个
截止2017-8-15日累计提问 2788个,站长回答了 2544个

上传截图