wait 加锁示例

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 start.

notify start.

notify end.

wait 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 start.

sleep end.

notify start.

notify end.

代码解析

从上述代码可以看出 sleep(1000) 方法(行号:11)执行之后,调用 notify() 方法并没有获取到 locker 锁,从上述执行结果中可以看出,而是执行完 sleep(1000) 方法之后才执行的 notify() 方法,因此可以证明调用 sleep() 方法并不会释放锁

知识扩展

1.sleep 和 wait 有什么区别?

sleep 和 wait 几乎是所有面试中必问的题,但想完全回答正确似乎没那么简单。

对于 sleep 和 wait 的区别,通常的回答是这样的:

  • wait 必须搭配 synchronize 一起使用,而 sleep 不需要;
  • 进入 wait 状态的线程能够被 notify 和 notifyAll 线程唤醒,而 sleep 状态的线程不能被 notify 方法唤醒;
  • wait 通常有条件地执行,线程会一直处于 wait 状态,直到某个条件变为真,但是 sleep 仅仅让你的线程进入睡眠状态;
  • wait 方法会释放对象锁,但 sleep 方法不会。

但上面的回答显然遗漏了一个重要的区别,在调用 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 的常见面试问题,希望本文可以帮助到你。

wait会释放锁吗_漫画:如何证明sleep不释放锁,而wait释放锁?相关推荐

  1. sql 闩锁 原因_如何识别和解决SQL Server中的热闩锁

    sql 闩锁 原因 描述 (Description) In SQL Server, internal latch architecture protects memory during SQL ope ...

  2. mysql innodb 锁类型_详细介绍MySQL InnoDB存储引擎各种不同类型的锁

    本文中,我们详细介绍MySQLInnoDB存储引擎各种不同类型的锁,以及不同SQL语句分别会加什么样的锁. 阅读提示 1.本文所参考的MySQL文档版本是8.0,做实验的MySQL版本是8.0.13 ...

  3. yield方法释放锁吗_死磕Synchronized底层实现重量级锁

    点击上方"Java知音",选择"置顶公众号" 技术文章第一时间送达! 作者:farmerjohngit 链接:https://github.com/farmer ...

  4. thread.sleep会释放锁吗_面试 LockSupport.park()会释放锁资源吗?

    (手机横屏看源码更方便) 引子 大家知道,我最近在招人,今天遇到个同学,他的源码看过一些,然后我就开始了AQS连环问. 我:说说AQS的大致流程? 他:AQS包含一个状态变量,一个同步队列--bala ...

  5. Redis 学习笔记-NoSQL数据库 常用五大数据类型 Redis配置文件介绍 Redis的发布和订阅 Redis_事务_锁机制_秒杀 Redis应用问题解决 分布式锁

    1.NoSQL数据库 1.1 NoSQL数据库概述 NoSQL(NosQL = Not Only sQL ),意即"不仅仅是sQL",泛指非关系型的数据库.NoSQL不依赖业务逻辑 ...

  6. redis分布式锁 在集群模式下如何实现_收藏慢慢看系列:简洁实用的Redis分布式锁用法...

    在微服务中很多情况下需要使用到分布式锁功能,而目前比较常见的方案是通过Redis来实现分布式锁,网上关于分布式锁的实现方式有很多,早期主要是基于Redisson等客户端,但在Spring Boot2. ...

  7. redistemplate分布式锁实现_基于 Redis SETNX 实现分布式锁

    环境与配置 Redis 任意版本即可 SpringBoot 任意版本即可,但是需要依赖 spring-boot-starter-data-redis <dependency><gro ...

  8. mysql死锁释放时间参数_由FTWRL导致的MySQL从库死锁分析及参数深究

    最近线上执行备份的从库时出现复制卡死现象,分析以后发现是两个死锁,show full processlist的状态如图1所示,其中,数据库版本是官方5.7.18版本,我们内部做了些许修改,但与此次死锁 ...

  9. 11. mysql锁机制_深入探讨MySQL锁机制

    MySQL锁机制究竟是怎样的呢?这是很多人都提到过的问题,下面就为您详细介绍MySQL锁机制方面的知识,希望可以让您MySQL锁机制有更多的了解. 当前MySQL已经支持 ISAM, MyISAM, ...

最新文章

  1. 当今主流分割网络有哪些?12篇文章一次带你看完
  2. python 生成器 迭代器 yiled
  3. oracle dataguard in-memory,Oracle 11g Dataguard 物理备库配置(一)之Duplicate配置
  4. [转]在Fedora上安装Oracle 11g XE
  5. RAID技术详细解答之一:入门基础篇
  6. [html] 页面布局时你使用最多的标签是什么?div吗?在什么情况下会使用到div?
  7. 数据结构--队列Queue--循环顺序队列
  8. # 根据三边求角度_七年级数学:怎么求旋转射线构成的角度?掌握这种方法口算出结果...
  9. 数据传输服务 DTS > 数据迁移 > 从自建数据库迁移至阿里云 > 源库为MySQL > 从自建MySQL迁移至RDS MySQL
  10. 编译器后端,寄存器分配算法
  11. 从一片空白到世界领先,中国通信翻身逆袭史
  12. 响应式网站设计 - 最佳实践
  13. iShot——Mac上功能最全的截图、录屏创造工具
  14. ECSHOP模板开发教程完全攻略
  15. Python根据字幕文件自动给视频添加字幕(通用版)
  16. 【单片机仿真项目】模拟开关灯(proteus原理图+keil代码)
  17. 阿里云大学-虚拟化技术入门-听课笔记
  18. 程序员如何自我认知 自我了解
  19. 计算机网络及工程实践西安电子科技有限公司,计算机网络-西安电子科技大学电子工程学院.PDF...
  20. 公司组织构架的三大类型

热门文章

  1. RHCE 学习笔记(5)- 本地用户和组的管理
  2. Tcpdump 超级详细的用法
  3. Part 1——使用 Go Modules
  4. 每天干攻防,都不会写驱动了
  5. web-13. 数组和字符串
  6. 【a202】【9208】输油管道问题
  7. Python学习——编码转换
  8. BLE - LINK LAYER SPECIFICATION
  9. Newtonsoft.Json 方法使用()
  10. 【C/C++】成员变量的初始化顺序