前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >多线程通信的三大法器,你真的会用吗?

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

作者头像
Java技术栈
发布2018-12-19 10:51:55
3840
发布2018-12-19 10:51:55
举报
文章被收录于专栏:Java技术栈

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 个区别》这篇文章。

实战

代码语言:javascript
复制
/**
 * 微信公众号: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

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2018-11-27,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Java技术栈 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 定义
  • 几个重要的点
  • 实战
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档