Java中wait和notify的简单使用

前言

在Java并发开发的过程中,我们总会遇到让一个线程等待另一个线程完成的案例。其实要实现这样的方式有很多,今天我主要给大家介绍的是怎么使用wait和notify实现这样一个案例。

简单介绍

  • wait() - 方法wait()的作用是使当前执行代码的线程进行等待,它是Object类的方法,该方法用来将当前线程置入预执行队列中,并且在wait所在的代码行处停止执行,直到接到通知或被中断为止。
    在调用wait方法之前,线程必须获得该对象的对象级别锁,即只能在同步方法或同步块中调用wait方法。

  • notify() - 同wait方法一样,也需要在同步方法或同步块中调用,即在调用前,线程也必须获得该对象的对象级别锁。

wait和notify调用时,如果没有持有适当的锁,将会抛出IllegalMonitorStateException的异常。它是一个RuntimeException的子类。

来个例子

为了模拟业务,我这里创建了三个类,NotifyThread、WaitThread和TestWaitNotify。其中NotifyThread和WaitThread为继承了Thread的线程,TestWaitNotify为执行这二个线程的调用类。具体如下:

NotifyThread.java

public class NotifyThread extends Thread {
    private final Object lock;

    public NotifyThread(Object lock) {
        super();
        this.lock = lock;
    }

    @Override
    public void run() {
        synchronized (lock) {
            // 做一些业务逻辑相关的事。。。。
            try {
                Thread.sleep(3000);
            } catch (Exception e) {
                e.printStackTrace();
            }
            // 业务逻辑完成了...
            System.out.println("开始 notify time= " + System.currentTimeMillis());
            lock.notify();
            System.out.println("结束 notify time= " + System.currentTimeMillis());
        }
    }
}

WaitThread.java类

public class WaitThread extends Thread {
    private final Object lock;

    public WaitThread(Object lock) {
        super();
        this.lock = lock;
    }

    @Override
    public void run() {
        try {
            synchronized (lock) {
                long start = System.currentTimeMillis();
                System.out.println("开始 wait time= " + start);
                lock.wait();
                long end = System.currentTimeMillis();
                System.out.println("结束 wait time= " + end);
                System.out.print("wait time = " + (end - start));
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

TestWaitNotify.java

public class TestWaitNotify {

    public static void main(String[] args) {
        Object lock = new Object();
        WaitThread t1 = new WaitThread(lock);
        t1.start();
        NotifyThread t2 = new NotifyThread(lock);
        t2.start();
    }
}

运行结果如下:


等待、通知机制的例子

总结

我们从上面的例子中可以看到,Wait和notify都被包裹在同步块synchronized中,而synchronized持有的对象锁都是lock。那如果它们持有的对象锁不一样会有什么样的结果呢?

扩展

我们把TestWaitNotify类改造成如下:

public class TestWaitNotify {

    public static void main(String[] args) {
        Object lock1 = new Object();
        Object lock2 = new Object();
        WaitThread t1 = new WaitThread(lock1);
        t1.start();
        NotifyThread t2 = new NotifyThread(lock2);
        t2.start();
    }
}

其结果如下;


wait、notify被包在持有不同对象锁的运行结果

NotifyThread线程运行3秒之后就已经结果了,而WaitThread却一直在等待中线程被唤醒或中断掉。


所以wait和notify所在的同步块或同步方法持有的对象锁是相同的情况下,才会有等待的线程被唤醒的可能。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。