并发编程学习(四):wait()、nitify()

1、wait、notify原理 

  •  Owner 线程发现条件不满足,调用wait()方法,即可进入WaitSet,变为WAITING 状态。
  • BLOCKED和WAITING的线程都处于阻塞状态,不占用CPU时间片。
  • BLOCKED线程会在Owner线程释放锁时唤醒。
  • WAITING线程会在Owner线程调用notify()或 notifyAll()时唤醒,但唤醒后并不意味着立刻获得锁,仍需进入EntryList重新竞争。

2、API介绍 

线程通信:wait() / notify() / notifyAll() 此三个方法定义在Object类中的。
  • obj.wait() 让进入object监视器的线程 到waitset等待。
  • obj.notify()在object上正在waitSet等待的线程挑一个唤醒。
  • obj.notifyAll()让object上正在waitSet等待的线程全部唤醒。

它们都是线程之间进行协作的手段,都属于Object对象的方法。必须获得此对象obj的锁,才能调用这几个方法

wait() 方法会释放对象的锁,进入WaitSet等待区(无限制等待,直到notify为止),从而然其他线程有机会获取到对象锁。

wait(long n)有时限的等待,到n毫秒后结束等待,或是被notify。 

未获得锁,直接调用wait()会报错。代码示例:

notify() / notifyAll()代码示例如下:

    final static Object obj = new Object();

    public static void main(String[] args) {

        new Thread(() -> {
            synchronized(obj) {
                log.debug("执行...");
                try {
                    obj.wait(); // 让线程在obj上一直等待下去
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                log.debug("其它代码...");
            }

        },"t1").start();

        new Thread(() -> {
            synchronized(obj) {
                log.debug("执行...");
                try {
                    obj.wait(); // 让线程在obj上一直等待下去
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                log.debug("其它代码...");
            }

        },"t2").start();

        // 主线程两秒后执行
        try {
            TimeUnit.SECONDS.sleep(2);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        log.debug("唤醒obj上的其它线程");
        synchronized (obj) {
//            obj.notify(); // 唤醒obj上随机一个线程
            obj.notifyAll(); // 唤醒obj上所有等待线程
        }

    }

obj.notify()代码执行结果:

14:15:50.663 [t1] DEBUG com.example.common.MagicNumberUtilsTests - 执行...
14:15:50.663 [t2] DEBUG com.example.common.MagicNumberUtilsTests - 执行...
14:15:52.673 [main] DEBUG com.example.common.MagicNumberUtilsTests - 唤醒obj上的其它线程
14:15:52.673 [t1] DEBUG com.example.common.MagicNumberUtilsTests - 其它代码...

obj.notifyAll()代码执行结果:

14:09:31.945 [t1] DEBUG com.example.common.MagicNumberUtilsTests - 执行...
14:09:31.949 [t2] DEBUG com.example.common.MagicNumberUtilsTests - 执行...
14:09:33.959 [main] DEBUG com.example.common.MagicNumberUtilsTests - 唤醒obj上的其它线程
14:09:33.959 [t2] DEBUG com.example.common.MagicNumberUtilsTests - 其它代码...
14:09:33.959 [t1] DEBUG com.example.common.MagicNumberUtilsTests - 其它代码...        

3、wait、notify的正确使用 

3.1、sleep(long n) 和wait(long n) 的区别

  1. sleep是Thread中的静态方法;而wait是Object中的方法。
  2. sleep不需要强制和synchronized配合使用;但wait需要和synchronized一起使用。
  3. sleep在睡眠的同时,不会释放对象锁;但wait在等待的时候会释放对象锁。
  4. 都调用这两个方法后,调用的当前线程都会进入:TIMED_WAITING状态。

3.2、正确使用格式

   final static Object lock = new Object();

// 干活线程
synchronized(lock) {
    while(条件不成立) {
        lock.wait();
    }
    // 干活
}

// 另一个线程
synchronized(lock) {
    lock.notifyAll();
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值