wait()、notify()、notifyAll()

wait()、notify()、notifyAll()这三个函数都是Object类中的方法,而Object类是所有类的父类,所以所有对象实例都有该方法.

wait():阻塞当前之前直到该对象(调用wait函数的对象)在另一个线程调用了notify()或者notifyAll();
notify():唤醒单个线程
notifyAll():唤醒所有线程

这三个方法,都是Java语言提供的实现线程间阻塞(Blocking)和控制进程内调度(inter-process communication)的底层机制。在解释如何使用前,先说明一下两点:

  1. 正如Java内任何对象都能成为锁(Lock)一样,任何对象也都能成为条件队列(Condition queue)。而这个对象里的wait(), notify()和notifyAll()则是这个条件队列的固有(intrinsic)的方法。

  2. 一个对象的固有锁和它的固有条件队列是相关的,为了调用对象X内条件队列的方法,你必须获得对象X的锁。这是因为等待状态条件的机制和保证状态连续性的机制是紧密的结合在一起的。

class TestWait {

    public static void main(String[]args) {
        TestWaitBean bean = new TestWaitBean("bean");
        System.out.println("init bean");
        System.out.println("invoke bean wait");
        try {
            bean.wait();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            System.out.println("wait finished");
        }
    }

    static class TestWaitBean {
        String name;
        public TestWaitBean(String name) {
            this.name = name;
        }
    }
}

上述代码运行后会抛出异常java.lang.IllegalMonitorStateException
因为未获取对象bean的锁,就去调用bean.wait()

修改一下代码,实现一个简单的阻塞、唤醒

class TestWait {

    public static void main(String[]args) {
        TestWaitBean bean = new TestWaitBean("bean");
        System.out.println("init bean");
        System.out.println("invoke bean wait");
        WeakUpThread thread = new WeakUpThread(bean);
        thread.start();
        synchronized (bean) {
            try {
                bean.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                System.out.println("wait finished");
            }
        }
    }

    static class TestWaitBean {
        String name;
        public TestWaitBean(String name) {
            this.name = name;
        }
    }


    static class WeakUpThread extends Thread {
        Object lock;
        public WeakUpThread(Object lock) {
            this.lock = lock;
        }
        @Override
        public void run() {
            super.run();
            synchronized (lock) {
                try {
                    System.out.println("Current Thread is sleep 2000ms" );
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {
                    System.out.println(" weakUp!!!");
                    lock.notify();
                }
            }
        }
    }
}

注意:

  • 不管是调用wait,还是notify、notifyAll,都要是在同步修饰的代码块或者方法中,即必须先获取对象锁,在调用对象方法

使用wait、notifyAll实现一个多线程的生产者、消费者

class BlockTest {

    public static void main(String []args) {
        List<Object>list = new ArrayList<>();
        Block block = new Block(list);
        list.add(null);

        Thread thread1 = new Thread(new PutThread(block, new Person("person1")), "thread1");
        Thread thread2 = new Thread(new PutThread(block, new Person("person2")), "thread2");
        Thread thread3 = new Thread(new PutThread(block, new Person("person3")), "thread3");
        Thread thread4 = new Thread(new PutThread(block, new Person("person4")), "thread4");
        Thread thread5 = new Thread(new OutThread(block), "thread5");
        Thread thread6 = new Thread(new OutThread(block), "thread6");
        Thread thread7 = new Thread(new OutThread(block), "thread7");

        thread1.start();
        thread2.start();
        thread5.start();
        thread6.start();
        thread7.start();
        thread3.start();
        thread4.start();
    }

    static class PutThread implements Runnable {
        Block block;
        Person person;

        public PutThread(Block block, Person person) {
            this.block = block;
            this.person = person;
        }
        @Override
        public void run() {
            System.out.println(Thread.currentThread().getName() + " puting the person name is " + person.getName());
            block.put(person);
        }
    }

    static class OutThread implements Runnable {
        Block block;
        public OutThread(Block block) {
            this.block = block;
        }
        @Override
        public void run() {
            Person  person = (Person) block.get();
            if(person != null) {
                System.out.println(Thread.currentThread().getName() + " get the person name is " + person.getName());
            } else {
                System.out.println(Thread.currentThread().getName() + " the person is null");
            }

        }
    }


    static class Block<T> {
        List<T>t;
        Object lock = new Object();
        int currentIndex = 0;
        volatile boolean isRead = false;
        public Block(List<T>t){
            this.t= t;
        }

        public void put(T at) {
            if(t == null) throw new NullPointerException("t is null");
            synchronized (lock) {
                try {
                    if(isRead) {
                        lock.wait();
                    }
                    currentIndex++;
                    t.add(at);
                    lock.notifyAll();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }

        public T get() {
            synchronized (lock){
                T at = null;
                try {
                    if(currentIndex == 0) {
                        System.out.println(Thread.currentThread().getName() +  " 当前下标已为0 阻塞 等待写入再取");
                        lock.wait();
                    }
                    if(isRead){
                        System.out.println(Thread.currentThread().getName() +  " 当前正在读 阻塞 等待写入再取 index" + currentIndex);
                        lock.wait();
                    }
                    isRead = true;
                    at = t.remove(currentIndex);
                    if(at == null) {
                        System.out.println(Thread.currentThread().getName() +  " index" + currentIndex);
                        return  null;
                    }
                    currentIndex --;
                    return at;
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {
                    lock.notifyAll();
                    isRead = false;
                    return at;
                }
            }
        }
    }

    static class Person{
        private String name = "";
        public Person(String name) {
            this.name = name;
        }
        public String getName() {
            return name;
        }
        public void setName(String name) {
            this.name = name;
        }
    }
}

*** 上述生产者、消费者只能适用于多个线程直接的put、get ***
因为wait、和notify/notifyAll本身就是互斥的,因为调用wait后,就会阻塞当前调用线程,
本身线程的notify/notifyAll也就不会被调用,所以说上述代码只能实现多个线程直接的场景,
还有就是上述代码只是简单的实现,而且是读操作加锁,一般而言应该是写操作时堵塞。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,080评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,422评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,630评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,554评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,662评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,856评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,014评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,752评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,212评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,541评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,687评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,347评论 4 331
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,973评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,777评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,006评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,406评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,576评论 2 349