一、简介

多个线程各自占有一些共享资源,并且互相等待其它线程占有的资源才能进行,而导致的两个或多个线程都在等待对方释放资源,都停止执行的情景。某一个同步块同时拥有“两个以上对象的锁”时,就可能会发生“死锁”的问题。

主要点:

  • 过多的同步可能造成相互不释放资源。
  • 从而互相等待,一般发生于同步中持有多个对象的锁。

解决:不要在同一个代码块中,同时持有多个对象的锁。

二、产生和解决死锁问题

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 + "涂口红");
            }
        }
    }
}