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公众号,关注后实时获知布最新的教程和优惠活动,谢谢。


问答区域    
2018-07-31 交作业
神乐



作业
public class TestFile {
    public static void main(String[] args) {
        final Hero a=new Hero("对象1",1);
        final Hero b=new Hero("对象2",1);
        final Hero c=new Hero("对象3",1);
        Thread thread1=new Thread(){
            public void run(){
                synchronized (a){
                    System.out.println("第一个线程已经占有对象1");
                    try{
                        Thread.sleep(1000);
                    }catch (InterruptedException e){
                        e.printStackTrace();
                    }
                    System.out.println("线程1试图占有线程2");
                    System.out.println("占有等待中...");
                    synchronized (b){
                        System.out.println("线程1 do something");
                    }
                    System.out.println("线程1试图占有线程3");
                    System.out.println("占有等待中...");
                    synchronized (c){
                        System.out.println("线程1 do something");
                    }
                }
            }
        };
        thread1.start();
        Thread thread2=new Thread(){
            public void run(){
                synchronized (b){
                    System.out.println("第二个线程已经占有对象2");
                    try {
                        Thread.sleep(1000);
                    }catch (InterruptedException e){
                        e.printStackTrace();
                    }
                    System.out.println("线程2试图占有对象1");
                    System.out.println("占有等待中...");
                    synchronized (a){
                        System.out.println("线程2 do something");
                    }
                    System.out.println("线程2试图占有对象3");
                    System.out.println("占有等待中...");
                    synchronized (c){
                        System.out.println("线程2 do something");
                    }
                }
            }
        };
        thread2.start();
        Thread thread3=new Thread(){
            public void run(){
                synchronized (c){
                    System.out.println("第三个线程已经占有对象3");
                    try {
                        Thread.sleep(1000);
                    }catch (InterruptedException e){
                        e.printStackTrace();
                    }
                    System.out.println("线程3试图占有对象1");
                    System.out.println("占有等待中...");
                    synchronized (a){
                        System.out.println("线程3 do something");
                    }
                    System.out.println("线程3试图占有对象2");
                    System.out.println("占有等待中...");
                    synchronized (b){
                        System.out.println("线程3 do something");
                    }
                }
            }
        };
        thread3.start();
    }
}

							






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





2018-07-06 哲学家就餐问题
TTLJ



典型的“哲学家就餐问题”: 哲学家围一圆桌吃饭,每人左右各一根筷子,但是要有两根筷子才能吃饭, 先让哲学家们拿起右手的筷子,这帮老家伙等1s后才想起拿左手的筷子,看看左边的都被上一个家伙当作“右手的筷子”拿起来。 然后就陷入江局了。
public class DeadLockTest {
	public static void main(String[] args) {
		final Chopstick c1 = new Chopstick("筷子1");
		final Chopstick c2 = new Chopstick("筷子2");
		final Chopstick c3 = new Chopstick("筷子3");
		
		Thread philosopher1 = new Thread() {
			public void run() {
				synchronized (c1) {
					System.out.println("哲学家1已占有" + c1);
					try {
                        //停顿1000毫秒,另一个线程有足够的时间占有安妮
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                    System.out.println("哲学家1 试图占有" + c3);
                    System.out.println("哲学家1 等待中 。。。。");
                    synchronized (c3) {
                        System.out.println("哲学家1获得" + c3);
                    }
				}
			}
		};
		Thread philosopher2 = new Thread() {
			public void run() {
				synchronized (c2) {
					System.out.println("哲学家2已占有" + c2);
					try {
                        //停顿1000毫秒,另一个线程有足够的时间占有安妮
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                    System.out.println("哲学家2 试图占有" + c1);
                    System.out.println("哲学家2 等待中 。。。。");
                    synchronized (c1) {
                        System.out.println("哲学家2获得" + c1);
                    }
				}
			}
		};
		Thread philosopher3 = new Thread() {
			public void run() {
				synchronized (c3) {
					System.out.println("哲学家3已占有" + c3);
					try {
                        //停顿1000毫秒,另一个线程有足够的时间占有安妮
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                    System.out.println("哲学家3 试图占有" + c1);
                    System.out.println("哲学家3 等待中 。。。。");
                    synchronized (c1) {
                        System.out.println("哲学家3获得" + c1);
                    }
				}
			}
		};
		

		philosopher1.start();
		philosopher2.start();
		philosopher3.start();
		
	}
}

class Chopstick {
	private String name;

	public String getName() {
		return name;
	}

	public void setChopstick(String name) {
		this.name = name;
	}

	public Chopstick(String name) {
		super();
		this.name = name;
	}

	@Override
	public String toString() {
		return getName();
	}
	
}

							






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





2018-05-27 分享一波
2018-05-08 和站长提供的答案不太一样,这样貌似也没啥问题
2018-05-08 和站长提供的答案不太一样,这样貌似也没啥问题
2018-05-08 照着demo写的,貌似没啥问题。
2018-04-23 三个对象的死锁里占有对象为什么一定要synchronized一个套一个?
2018-04-22 最下面那个同志提的问题,确实不是同步对象,但我加了final还是死锁不了,为什么?那么该如何保证同步对象呢?
2017-11-03 时间啊
2016-07-10 代码没实现死锁,不知道问题在哪




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

尽量提供截图代码异常信息,有助于分析和解决问题。 也可进本站QQ群交流: 620943819
提问尽量提供完整的代码,环境描述,越是有利于问题的重现,您的问题越能更快得到解答。
对教程中代码有疑问,请提供是哪个步骤,哪一行有疑问,这样便于快速定位问题,提高问题得到解答的速度
在已经存在的几千个提问里,有相当大的比例,是因为使用了和站长不同版本的开发环境导致的,比如 jdk, eclpise, idea, mysql,tomcat 等等软件的版本不一致。
请使用和站长一样的版本,可以节约自己大量的学习时间。 站长把教学中用的软件版本整理了,都统一放在了这里, 方便大家下载: http://how2j.cn/k/helloworld/helloworld-version/1718.html

上传截图