可以看到,notify()
和 notifyAll()
的作用是一致的,都是去唤醒等待中的线程。但是也正如他们方法名所描述的,notify()
会"随机"唤醒一个等待线程,而notifyAll()
会尝试唤醒所有的等待中的线程。
注意这里的唤醒,并不是真的唤醒去执行,实际上只是让处于等待的线程,有重新获取锁的争抢权,也就是说,哪怕此时有一百个线程处于等待状态,此时调用notifyAll()
也只会有一个线程获取到锁,允许进入临界区执行。
这在底层中,其实是利用了两个等待队列来实现的,分别是入口队列(EntrySet)和等待队列(WaitSet)。
被 Synchronized 阻塞等待的线程,会进入入口队列,而当条件不满足时,主动调用 wait()
方法进入等待的线程,则会进入等待队列。
在等待队列中的线程,如果不被唤醒,则永远没有锁的争抢权,无法获取锁也就无法被执行。
2.2 为什么说尽量使用 notifyAll?
终于进入主题了,就前面的描述,看似应该是使用 notify()
更好一些。因为即便我们通知了等待队列中,所有的线程,但同一时刻,也只有一个线程可以获取互斥锁,进入临界区执行,这么看来 notify()
会更高效一些。
但是这里埋下来一个风险,就是只使用 notify() 可能会导致某些线程,一直处于等待队列中,而永远不会被唤醒并获得执行权。
理想情况下,一次等待(wait)对应一次通知(notify),是非常完美的,但是实际业务场景下,可能做不到。
举个例子,在多生产者消费者模式下,待处理的数据队列只有一条数据了,理想场景下,消费者在处理掉一条数据后,理论上应该唤醒生产者再生产一条新的待消费数据。可是 notify()
是随机唤醒,也就是它可能会唤醒一个消费者线程,这个消费者线程,发现没有待处理的数据,此时条件不满足,又主动进入等待队列。
也正是因为如此,在并发编程中有个范式模板:
synchronized(this){ while(条件不满足) { wait(); } // … }
这段代码,大家应该很熟悉,notify()
只能保证唤醒一个线程,但是不保证线程执行的时候,曾经的等待条件已经被满足了。为了保证可靠性,此处使用循环检测的方式,只有必要条件满足时,才继续执行。
正是因为 notify()
随机唤醒的特点,导致在多条件的情况下,会导致某些线程永远不会被通知到。稳妥的方式,是使用 notifyAll()
,让等待中的线程,都有一次再执行的权利。
这也就是为什么说,除非深思熟虑,否则尽量使用 notifyAll()。
2.3 什么是深思熟虑?
使用 notifyAll()
主要是为了稳定,减少程序的复杂度,我们程序员,解决的是一系列工程问题,虽然有时候需要挑战一些性能的极限,但是大多数时候应该是以稳定且易读易维护为出发点实现功能。
但是在程序的世界中,永远没有绝对的银弹。不带场景去分析问题,都是耍流氓。
前面说 “除非深思熟虑”,那什么场景下才可以用到深思熟虑?
其实只要满足三个条件即可:
-
线程进入等待队列的条件相同。
-
在满足条件时,所有线程执行的逻辑相同。
-
当一个线程执行完(无论是否异常),必定唤醒一个线程。
怎么理解呢?
当所有的线程,执行相同的业务逻辑,那么触发等待的条件也必定相同,只要保证一个线程在退出的时候,必定调用 notify()
唤醒一个线程。满足这些条件,就可以只使用 notify()
方法。
生产者消费者模式,必定是不满足的,它把线程分为两个性质,做着不同的任务,等着着不同的条件。但是这种情况下,使用 Lock&Condition 会更适合。
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
Android高级架构师
由于篇幅问题,我呢也将自己当前所在技术领域的各项知识点、工具、框架等汇总成一份技术路线图,还有一些架构进阶视频、全套学习PDF文件、面试文档、源码笔记。
- 330页PDF Android学习核心笔记(内含上面8大板块)
-
Android学习的系统对应视频
-
Android进阶的系统对应学习资料
- Android BAT部分大厂面试题(有解析)
好了,以上便是今天的分享,希望为各位朋友后续的学习提供方便。觉得内容不错,也欢迎多多分享给身边的朋友哈。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
好了,以上便是今天的分享,希望为各位朋友后续的学习提供方便。觉得内容不错,也欢迎多多分享给身边的朋友哈。