Java 多线程 --死锁及解决方案

在java 多线程中 过多的同步造成相互不释放资源 从而相互等待,造成死锁线现象,一般发生于同步中持有多个对象锁
如以下代码:

public class DeadLock {
	public static void main(String[] args) {
		WuDao wd1 = new WuDao(0, "金毛狮王");
		WuDao wd2 = new WuDao(1, "灭绝师太");
		wd1.start();
		wd2.start();
	}
}

// 屠龙刀
class TuLongDao {

}

// 倚天剑
class YiTianJian {

}

class WuDao extends Thread {
	static TuLongDao tld = new TuLongDao();
	static YiTianJian ytj = new YiTianJian();
	// 选择
	int choice;
	// 名字
	String name;

	@Override
	public void run() {
		// 悟道
		wudao();
	}

	public WuDao(int choice, String name) {
		super();
		this.choice = choice;
		this.name = name;
	}

	private void wudao() {
		if (choice == 0) {
			synchronized (tld) { // 获得屠龙刀锁
				System.out.println(this.name + "领悟屠龙刀法");
				// 1秒后想拥有倚天剑锁
				try {
					Thread.sleep(1000);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
				synchronized (ytj) {
					System.out.println(this.name + "领悟武穆遗书");
				}
			}
		} else {
			synchronized (ytj) { // 获得倚天锁
				System.out.println(this.name + "领悟倚天剑法");
				// 2秒后想拥有屠龙刀锁
				try {
					Thread.sleep(2000);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
				synchronized (tld) {
					System.out.println(this.name + "领悟九阴真经");
				}
			}
		}
	}
}

运行以上代码:
在这里插入图片描述

可以看到:由于发生死锁,程序不能继续运行

解决方案:
不要在同一代码块中,同时持有多个对象锁。
我们只要把synchronized 代码块中的第二个锁往外移一下就可以了,修改后代码如下:

private void wudao() {
		if (choice == 0) {
			synchronized (tld) { // 获得屠龙刀锁
				System.out.println(this.name + "领悟屠龙刀法");
				// 1秒后想拥有倚天剑锁
				try {
					Thread.sleep(1000);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
				/*synchronized (ytj) {
					System.out.println(this.name + "领悟武穆遗书");
				}*/
			}
			synchronized (ytj) {
				System.out.println(this.name + "领悟武穆遗书");
			}
		} else {
			synchronized (ytj) { // 获得倚天锁
				System.out.println(this.name + "领悟倚天剑法");
				// 2秒后想拥有屠龙刀锁
				try {
					Thread.sleep(2000);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
				/*synchronized (tld) {
					System.out.println(this.name + "领悟九阴真经");
				}*/
			}
			synchronized (tld) {
				System.out.println(this.name + "领悟九阴真经");
			}
		}
	}

再次运行修改后代码:
在这里插入图片描述
代码可以正常运行了。

posted @ 2020-04-01 02:02  行者老夫  阅读(776)  评论(0编辑  收藏  举报