美文网首页
JAVA多线程之wait/notify

JAVA多线程之wait/notify

作者: 不知名的蛋挞 | 来源:发表于2018-10-25 09:36 被阅读16次

    原文:https://www.cnblogs.com/hapjin/p/5492645.html

    (1)wait() 与 notify/notifyAll 方法必须在同步代码块中使用

    wait() 与 notify/notifyAll() 是Object类的方法,在执行两个方法时,要先获得锁。那么怎么获得锁呢?使用synchronized关键字获得锁。因此,wait() 与 notify/notifyAll() 经常与synchronized搭配使用,即在synchronized修饰的同步代码块或方法里面调用wait() 与 notify/notifyAll()方法。

    (2)wait() 与 notify/notifyAll() 的执行过程

    由于 wait() 与 notify/notifyAll() 是放在同步代码块中的,因此线程在执行它们时,肯定是进入了临界区中的,即该线程肯定是获得了锁的。

    • 当线程执行wait()时,会把当前的锁释放,然后让出CPU,进入等待状态。
    • 当执行notify/notifyAll方法时,会唤醒一个处于等待该 对象锁 的线程,然后继续往下执行,直到执行完退出对象锁锁住的区域(synchronized修饰的代码块)后再释放锁。

    从这里可以看出,notify/notifyAll()执行后,并不立即释放锁,而是要等到执行完临界区中代码后,再释放。故,在实际编程中,我们应该尽量在线程调用notify/notifyAll()后,立即退出临界区。即不要在notify/notifyAll()后面再写一些耗时的代码。示例如下:

    public class Service {
    
        public void testMethod(Object lock) {
            try {
                synchronized (lock) {
                    System.out.println("begin wait() ThreadName="
                            + Thread.currentThread().getName());
                    lock.wait();
                    System.out.println("  end wait() ThreadName="
                            + Thread.currentThread().getName());
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    
        public void synNotifyMethod(Object lock) {
            try {
                synchronized (lock) {
                    System.out.println("begin notify() ThreadName="
                            + Thread.currentThread().getName() + " time="
                            + System.currentTimeMillis());
                    lock.notify();
                    Thread.sleep(5000);
                    System.out.println("  end notify() ThreadName="
                            + Thread.currentThread().getName() + " time="
                            + System.currentTimeMillis());
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
    

    在第3行的testMethod()中调用 wait(),在第17行的synNotifyMethod()中调用notify()。

    从上面的代码可以看出,wait() 与 notify/notifyAll()都是放在同步代码块中才能够执行的。如果在执行wait() 与 notify/notifyAll() 之前没有获得相应的对象锁,就会抛出:java.lang.IllegalMonitorStateException异常。

    在第8行,当ThreadA线程执行lock.wait();这条语句时,释放获得的对象锁lock,并放弃CPU,进入等待队列。

    另一个线程执行第23行lock.notify();,会唤醒ThreadA,但是此时它并不立即释放锁,接下来它睡眠了5秒钟(sleep()是不释放锁的,事实上sleep()也可以不在同步代码块中调用),直到第28行,退出synchronized修饰的临界区时,才会把锁释放。这时,ThreadA就有机会获得另一个线程释放的锁,并从等待的地方起(第9行)起开始执行。

    接下来是两个线程类,线程类ThreadA调用testMethod()方法执行lock.wait();时被挂起,另一个线程类synNotifyMethodThread调用synNotifyMethod()负责唤醒挂起的线程。代码如下:

    public class ThreadA extends Thread {
        private Object lock;
    
        public ThreadA(Object lock) {
            super();
            this.lock = lock;
        }
    
        @Override
        public void run() {
            Service service = new Service();
            service.testMethod(lock);
        }
    }
    
    public class SynNotifyMethodThread extends Thread {
        private Object lock;
    
        public SynNotifyMethodThread(Object lock) {
            super();
            this.lock = lock;
        }
    
        @Override
        public void run() {
            Service service = new Service();
            service.synNotifyMethod(lock);
        }
    }
    

    再接下来是测试类:

    public class Test {
    
        public static void main(String[] args) throws InterruptedException {
    
            Object lock = new Object();
    
            ThreadA a = new ThreadA(lock);
            a.start();
    
            SynNotifyMethodThread c = new SynNotifyMethodThread(lock);
            c.start();
        }
    }
    

    (3)notify 通知的顺序不能错

    假设在线程A中执行wait(),在线程B中执行notify()。但如果线程B先执行了notify()然后结束了,线程A才去执行wait(),那此时,线程A将无法被正常唤醒了。

    相关文章

      网友评论

          本文标题:JAVA多线程之wait/notify

          本文链接:https://www.haomeiwen.com/subject/msfqtqtx.html