一、简介
多个线程各自占有一些共享资源,并且互相等待其它线程占有的资源才能进行,而导致的两个或多个线程都在等待对方释放资源,都停止执行的情景。某一个同步块同时拥有“两个以上对象的锁”时,就可能会发生“死锁”的问题。
主要点:
- 过多的同步可能造成相互不释放资源。
- 从而互相等待,一般发生于同步中持有多个对象的锁。
解决:不要在同一个代码块中,同时持有多个对象的锁。
二、产生和解决死锁问题
1、定义量
//口红
class LipStick {
}
//镜子
class Mirror {
}
2、定义化妆类,调用产生死锁代码
//化妆
class Markup extends Thread {
static LipStick lipStick = new LipStick();
static Mirror mirror = new Mirror();
//选择
int choice;
// 名字
String girl;
public Markup(int choice, String girl) {
this.choice = choice;
this.girl = girl;
}
@Override
public void run() {
// 化妆
markup();
}
3、死锁的产生位置
// 相互持有对方的对象锁--》可能造成死锁
private void markup() {
if (choice == 0) {
synchronized (lipStick) {//获得口红的锁
System.out.println(this.girl + "涂口红");
// 1秒后想拥有镜子的锁
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
// 产生死锁
synchronized (lipStick){
System.out.println(this.girl+"涂口红");
}
}
} else {
synchronized (mirror) {//获得镜子的锁
System.out.println(this.girl + "照镜子");
// 1秒后想拥有口红的锁
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
// 产生死锁
synchronized (lipStick){
System.out.println(this.girl+"涂口红");
}
}
}
}
4、死锁的解决
将获取锁的代码往外面移动一个,使先完成照镜子的动作后,进入线程等待(等待获取口红的状态)。
// 解决死锁问题
synchronized (lipStick) {
System.out.println(this.girl + "涂口红");
}
5、调用方法
public static void main(String[] args) {
Markup g1 = new Markup(1, "张柏芝");
Markup g2 = new Markup(0, "张柏芝");
g1.start();
g2.start();
}
三、完整代码
public class DeadLock {
public static void main(String[] args) {
Markup g1 = new Markup(1, "张柏芝");
Markup g2 = new Markup(0, "张柏芝");
g1.start();
g2.start();
}
}
//口红
class LipStick {
}
//镜子
class Mirror {
}
//化妆
class Markup extends Thread {
static LipStick lipStick = new LipStick();
static Mirror mirror = new Mirror();
//选择
int choice;
// 名字
String girl;
public Markup(int choice, String girl) {
this.choice = choice;
this.girl = girl;
}
@Override
public void run() {
// 化妆
markup();
}
// 相互持有对方的对象锁--》可能造成死锁
private void markup() {
if (choice == 0) {
synchronized (lipStick) {//获得口红的锁
System.out.println(this.girl + "涂口红");
// 1秒后想拥有镜子的锁
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
// 产生死锁
/**
synchronized (lipStick){
System.out.println(this.girl+"涂口红");
}
*/
}
// 解决死锁问题
synchronized (lipStick) {
System.out.println(this.girl + "涂口红");
}
} else {
synchronized (mirror) {//获得镜子的锁
System.out.println(this.girl + "照镜子");
// 1秒后想拥有口红的锁
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
// 产生死锁
/**
synchronized (lipStick){
System.out.println(this.girl+"涂口红");
}
*/
}
// 解决死锁问题
synchronized (lipStick) {
System.out.println(this.girl + "涂口红");
}
}
}
}