wait, notify, notifyAll 是多线程之间通信最重要的 3 个方法,今天,栈长给大家普及一下它们的知识要点及应用实战。
定义
wait:让持有该对象锁的线程等待;
notify: 唤醒任何一个持有该对象锁的线程;
notify: 唤醒所有持有该对象锁的线程;
它们 3 个的关系是,调用对象的 wait 方法使线程暂停运行,通过 notify/ notifyAll 方法唤醒调用 wait 暂时的线程。
然而,它们并不是 Thread 类中的方法,而是 Object 类中的,为什么呢!? 因为每个对象都有监视锁,线程要操作某个对象当然是要获取某个对象的锁了,而不是线程的锁。
image
如图所示,wait 带时间表示最大超时时间,过了时间还不唤醒就会自动唤醒线程重新竞争对象锁。
几个重要的点
1、调用对象的 wait, notify, notifyAll 方法需要拥有对象的监视器锁,即它们只能在同步方法(块)中使用;
2、调用 wait 方法会使用线程暂停并让出 CPU 资源,同时释放持有的对象的锁;
3、多线程使用 notify 容易发生死锁,一般使用 notifyAll;
4、关于 wait 和 sleep 的详细区别请翻阅 《多线程 sleep 和 wait 的 5 个区别》这篇文章。
实战
/** * 微信公众号:Java技术栈 */public static void main(String[] args) { Object lock = new Object(); Thread t1 = new Thread(() -> { synchronized (lock) { for (int i = 0; i < 20; i++) { System.out.print(i); if (i == 10) { try { lock.wait(); } catch (InterruptedException e) { e.printStackTrace(); } } } } }); Thread t2 = new Thread(() -> { synchronized (lock) { try { Thread.sleep(5000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.print("Java技术栈"); lock.notifyAll(); } }); t1.start(); t2.start(); }
上面的例子结合 wait/ notifyAll 来演示了它们的相互作用。
线程 t1 首先输出 012345678910
,5秒后继续输出 Java技术栈111213141516171819
。
作者:Java技术栈
链接:https://www.jianshu.com/p/3aa15bd06a5c
点击查看更多内容
为 TA 点赞
评论
共同学习,写下你的评论
评论加载中...
作者其他优质文章
正在加载中
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦