Java并发:await/wait的区别

在这里插入图片描述
在使用Lock之前,我们都使用Object 的wait和notify实现同步的。举例来说,一个producer和consumer,consumer发现没有东西了,等待,produer生成东西了,唤醒。

在这里插入图片描述

有了lock后,世道变了,现在是:
在这里插入图片描述
为了突出区别,省略了若干细节。区别有三点:

  1. lock不再用synchronize把同步代码包装起来;
  2. 阻塞需要另外一个对象condition;
  3. 同步和唤醒的对象是condition而不是lock,对应的方法是await和signal,而不是wait和notify。
    为什么需要使用condition呢?简单一句话,lock更灵活。以前的方式只能有一个等待队列,在实际应用时可能需要多个,比如读和写。为了这个灵活性,lock将同步互斥控制和等待队列分离开来,互斥保证在某个时刻只有一个线程访问临界区(lock自己完成),等待队列负责保存被阻塞的线程(condition完成)。

通过查看ReentrantLock的源代码发现,condition其实是等待队列的一个管理者,condition确保阻塞的对象按顺序被唤醒。

1.object.wait()

使用方法:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
假设三个线程执行顺序

线程A–>线程C–>线程B //没毛病,因为wait后是释放了锁的

所以问题来了:等待的线程中有A和C, B notify后,只会唤醒其中一个执行(notifyAll同样只有一个执行);假如我们的需求是想让A线程执行,那么这种object的方式是无法控制的

2.所以condition来了

使用方法:注意condition是依赖ReentrantLock
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
所以通过condition与object进行线程通信的区别已经很明显了,condition更加灵活。

个人理解,本质上来讲:

多线程环境的下,线程直接的互斥[执行]依靠的应该是锁Lock,线程的之间的[通信]依靠的应该是条件Condition/信号,一般情况下lock确实可以同时满足做这两个事情,所以在Object的方式满足了这个一般情况,但是肯定会有复杂的场景比如刚才例子中,需要让满足一定条件的线程执行,仅仅依靠锁是不能完美解决的。所以condition实际上分离了执行和通信。

总结:
synchronized :等待wait(), 唤醒随机一个线程:notify() ,唤醒所有线程:notifyAll()
ReentrantLock:等待await(), 唤醒线程:signal()

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值