本文转载自微信公众号「Java中文社群」,作者磊哥。转载本文请联系Java中文社群公众号。
- public class WaitDemo {
- private static Object locker = new Object();
- public static void main(String[] args) throws InterruptedException {
- WaitDemo waitDemo = new WaitDemo();
- // 启动新线程,防止主线程被休眠
- new Thread(() -> {
- try {
- waitDemo.doWait();
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- }).start();
- Thread.sleep(200); // 此行本身没有意义,是为了确保 wait() 先执行再执行 notify()
- waitDemo.doNotify();
- }
- /**
- * 执行 wait()
- */
- private void doWait() throws InterruptedException {
- synchronized (locker) {
- System.out.println("wait start.");
- locker.wait();
- System.out.println("wait end.");
- }
- }
- /**
- * 执行 notify()
- */
- private void doNotify() {
- synchronized (locker) {
- System.out.println("notify start.");
- locker.notify();
- System.out.println("notify end.");
- }
- }
- }
以上程序的执行结果为:
代码解析
从上述代码可以看出,我们给 wait() 和 notify() 两个方法上了同一把锁(locker),但在调用完 wait() 方法之后 locker 锁就被释放了,所以程序才能正常执行 notify() 的代码,因为是同一把锁,如果不释放锁的话,是不会执行 notify()的代码的,这一点也可以从打印的结果中证实(结果输出顺序),所以综合以上情况来说 wait() 方法是释放锁的。
sleep 加锁示例
- public class WaitDemo {
- private static Object locker = new Object();
- public static void main(String[] args) throws InterruptedException {
- WaitDemo waitDemo = new WaitDemo();
- // 启动新线程,防止主线程被休眠
- new Thread(() -> {
- synchronized (locker) {
- try {
- System.out.println("sleep start.");
- Thread.sleep(1000);
- System.out.println("sleep end.");
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- }
- }).start();
- Thread.sleep(200);
- waitDemo.doNotify();
- }
- /**
- * 执行 notify()
- */
- private void doNotify() {
- synchronized (locker) {
- System.out.println("notify start.");
- locker.notify();
- System.out.println("notify end.");
- }
- }
- }
以上程序的执行结果为:
代码解析
从上述代码可以看出 sleep(1000) 方法(行号:11)执行之后,调用 notify() 方法并没有获取到 locker 锁,从上述执行结果中可以看出,而是执行完 sleep(1000) 方法之后才执行的 notify() 方法,因此可以证明调用 sleep() 方法并不会释放锁。
知识扩展
1.sleep 和 wait 有什么区别?
sleep 和 wait 几乎是所有面试中必问的题,但想完全回答正确似乎没那么简单。
对于 sleep 和 wait 的区别,通常的回答是这样的:
但上面的回答显然遗漏了一个重要的区别,在调用 wait 方法之后,线程会变为WATING 状态,而调用 sleep 方法之后,线程会变为 TIMED_WAITING 状态。
2.wait 能不能在 static 方法中使用?为什么?
不能,因为 wait 方法是实例方法(非 static 方法),因此不能在 static 中使用,源码如下:
- public final void wait() throws InterruptedException {
- wait(0);
- }
3.wait/notify 可以不搭配 synchronized 使用吗?为什么?
不行,因为不搭配 synchronized 使用的话程序会报错,如下图所示:
更深层次的原因是因为不加 synchronized 的话会造成 Lost Wake-Up Problem,唤醒丢失的问题,详情可见:https://juejin.im/post/5e6a4d8a6fb9a07cd80f36d1
总结
本文我们通过 synchronized 锁定同一对象,来测试 wait 和 sleep 方法,再通过执行结果的先后顺序证明:wait 方法会释放锁,而 sleep 方法并不会。同时我们还讲了几个 wait 和 sleep 的常见面试问题,希望本文可以帮助到你。
原文链接:原文链接:https://mp.weixin.qq.com/s/eYijPq_PtMd93dsMDUPNLQ
我从没参加过一场得喊着和对方说话的「发布会」。 不过,与其说是发布会,苹果为...
拖着疲惫的身体回到家,辛苦工作了一天的打工人,终于开始了翻身农奴把歌唱的生...
据报道,微软目前正在研发一款聊天机器人,如果未来成功投入使用,那么,通过数...
作为国内的即时聊天工具,微信经过这么多年的发展,俨然已经成为了全民级的移动...
2019年,21岁的中国学生李凡在自己的个人微博上留下一行话后服药自杀。 根据后续...
如今,人工智能技术已经广泛应用在各行业和领域:可穿戴设备、汽车、生产力应用...
早在去年年底,业内就普遍分析,2020年全球手机市场将面临巨大的不确定性:一方...
微信苹果二选一,成为近期网络上热门的话题。 此事件源自最新微信禁令,这让苹果...
1.眼睛是相机,眨眨眼就是想把你拍下来存在脑子里。 2.我本就是薄凉之人,却为...
用户画像作为一种设计工具,可以很好地帮助设计师跳出为自己设计的惯性思维,聚...