漫画:如何证明sleep不释放锁,而wait释放锁?

简介: 漫画:如何证明sleep不释放锁,而wait释放锁?

image.png

image.png


image.png


image.png


image.png


image.png


image.png


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 有什么区别?


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


对于 sleepwait 的区别,通常的回答是这样的:


  • 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 使用的话程序会报错,如下图所示:


image.png


更深层次的原因是因为不加 synchronized 的话会造成 Lost Wake-Up Problem,唤醒丢失的问题,详情可见:https://juejin.im/post/5e6a4d8a6fb9a07cd80f36d1


总结


本文我们通过 synchronized 锁定同一对象,来测试 waitsleep 方法,再通过执行结果的先后顺序证明:wait 方法会释放锁,而 sleep 方法并不会。同时我们还讲了几个 waitsleep 的常见面试问题,希望本文可以帮助到你。

相关文章
|
JSON 网络协议 数据格式
curl常用参数详解及示例
curl是一个开源的命令行工具,它基于网络协议,对指定URL进行网络传输,得到数据后不任何具体处理(如:html的渲染等),直接显示在"标准输出"(stdout)上。
3813 1
|
消息中间件 运维 Java
【Log日志】logback.xml动态配置属性值(包括接入的第三方配置)
1如何动态配置Logback的存放路径 我们在开发过程中,会使用到logback.xml 配置来管理日志文件; 比如
3388 0
|
3月前
|
前端开发 搜索推荐 开发者
深入理解 HTML 中的<h1>标签:语义、用法与最佳实践
本文深入解析HTML中<h1>标签的核心特性、使用规范及常见误区,探讨其在SEO优化、页面结构与可访问性中的重要作用,帮助开发者构建语义清晰、结构合理的网页文档。
359 1
|
缓存 JavaScript Java
常见java OOM异常分析排查思路分析
Java虚拟机(JVM)遇到内存不足时会抛出OutOfMemoryError(OOM)异常。常见OOM情况包括:1) **Java堆空间不足**:大量对象未被及时回收或内存泄漏;2) **线程栈空间不足**:递归过深或大量线程创建;3) **方法区溢出**:类信息过多,如CGLib代理类生成过多;4) **本机内存不足**:JNI调用消耗大量内存;5) **GC造成的内存不足**:频繁GC但效果不佳。解决方法包括调整JVM参数(如-Xmx、-Xss)、优化代码及使用高效垃圾回收器。
647 15
常见java OOM异常分析排查思路分析
|
搜索推荐 算法 安全
程序化广告系列之一---名词解释
这篇文章是关于程序化广告中各种专业术语的详细解释,包括DSP、SSP、RTB等,以及它们在广告交易流程中的作用和关系。
1246 2
程序化广告系列之一---名词解释
|
Windows
LabVIEW强制重新安装无法运行或损坏的NI软件
LabVIEW强制重新安装无法运行或损坏的NI软件
217 1
|
物联网 Android开发
【Android App】低功耗蓝牙中扫描BLE设备的讲解及实战(附源码和演示 超详细)
【Android App】低功耗蓝牙中扫描BLE设备的讲解及实战(附源码和演示 超详细)
1150 0
|
SQL 监控 算法
|
前端开发 UED
探索前端工程化之路:Webpack、Rollup等构建工具对比与实践
在现代前端开发中,工程化成为不可或缺的一环。本文将深入探讨常用的前端构建工具Webpack和Rollup,并比较它们在实践中的优劣势。通过对功能、性能、插件生态等方面的评估,帮助读者选择适合自己项目需求的构建工具。
485 1
|
SQL 消息中间件 存储
Flink SQL 实战:HBase 的结合应用
本文着重介绍 HBase 和 Flink 在实际场景中的结合使用。主要分为两种场景,第一种场景:HBase 作为维表与 Flink Kafka table 做 temporal table join 的场景;第二种场景:Flink SQL 做计算之后的结果写到 HBase 表,供其他用户查询的场景。
Flink SQL 实战:HBase 的结合应用