多线程通信的三大法器,你真的会用吗?

发布时间:2019-11-19 发布网站:脚本宝典
脚本宝典收集整理的这篇文章主要介绍了多线程通信的三大法器,你真的会用吗?脚本宝典觉得挺不错的,现在分享给大家,也给大家做个参考。

wait, notify, notifyAll 是多线程之间通信最重要的 3 个方法,今天,栈长给大家普及一下它们的知识要点及应用实战。

定义

waIT:让持有该对象锁的线程等待;

notify: 唤醒任何一个持有该对象锁的线程;

notify: 唤醒所有持有该对象锁的线程;

它们 3 个的关系是,调用对象的 wait 方法使线程暂停运行,通过 notify/ notifyAll 方法唤醒调用 wait 暂时的线程。

然而,它们并不是 Thread 类中的方法,而是 Object 类中的,为什么呢!? 因为每个对象都有监视锁,线程要操作某个对象当然是要获取某个对象的锁了,而不是线程的锁。

多线程通信的三大法器,你真的会用吗?

如图所示,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 多线程技术文章请在Java技术栈微信公众号后台回复关键字:多线程。

本文原创首发于微信公众号:Java技术栈(id:javastack),关注公众号在后台回复 "多线程" 可获取更多,转载请原样保留本信息。

多线程通信的三大法器,你真的会用吗?

脚本宝典总结

以上是脚本宝典为你收集整理的多线程通信的三大法器,你真的会用吗?全部内容,希望文章能够帮你解决多线程通信的三大法器,你真的会用吗?所遇到的问题。

如果觉得脚本宝典网站内容还不错,欢迎将脚本宝典推荐好友。

本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。
如您有任何意见或建议可联系处理。小编QQ:384754419,请注明来意。