漫画:怎么证明Sleep不释放锁,而Wait释放锁?

网站建设5年前发布
317 0 0

20230305210922b6ad2bd4590e8fa442b871fb105e75e9039f85814,2023030521090104db713624b15ced615300b5f5922b85488b8b553,202303052109227440816702ccc36a295242eebba22067cec573222,2023030521090229025b065747df7eaa4098b3485053ee42acb1792,20230305210905c6ca565233a617e55984077ce3701de945d94d232,2023030521090474a16bd2999ff7c164b414080c1496de4897d6380,20230305210905c75509b63300cc77e52610b94a90f9a973c379405,以上程序的执行结果为:,代码解析,从上述代码可以看出,我们给 wait() 和 notify() 两个方法上了同一把锁(locker),但在调用完 wait() 方法之后 locker 锁就被释放了,所以程序才能正常执行 notify() 的代码,因为是同一把锁,如果不释放锁的话,是不会执行 notify() 的代码的,这一点也可以从打印的结果中证实(结果输出顺序),所以综合以上情况来说 wait() 方法是释放锁的。,以上程序的执行结果为:,从上述代码可以看出 sleep(1000) 方法(行号:11)执行之后,调用 notify() 方法并没有获取到 locker 锁,从上述执行结果中可以看出,而是执行完 sleep(1000) 方法之后才执行的 notify() 方法,因此可以证明调用 sleep() 方法并不会释放锁。,sleep 和 wait 几乎是所有面试中必问的题,但想完全回答正确似乎没那么简单。,对于 sleep 和 wait 的区别,通常的回答是这样的:,wait 必须搭配 synchronize 一起使用,而 sleep 不需要;,进入 wait 状态的线程能够被 notify 和 notifyAll 线程唤醒,而 sleep 状态的线程不能被 notify 方法唤醒;,wait 通常有条件地执行,线程会一直处于 wait 状态,直到某个条件变为真,但是 sleep 仅仅让你的线程进入睡眠状态;,wait 方法会释放对象锁,但 sleep 方法不会。,但上面的回答显然遗漏了一个重要的区别,在调用 wait 方法之后,线程会变为 WATING 状态,而调用 sleep 方法之后,线程会变为 TIMED_WAITING 状态。,不能,因为 wait 方法是实例方法(非 static 方法),因此不能在 static 中使用,源码如下:,不行,因为不搭配 synchronized 使用的话程序会报错,如下图所示:,20230305210907354ca0f4035491ba6a9460e821a4c740cb8a80454,更深层次的原因是因为不加 synchronized 的话会造成 Lost Wake-Up Problem,唤醒丢失的问题,详情可见:https://juejin.im/post/5e6a4d8a6fb9a07cd80f36d1,本文我们通过 synchronized 锁定同一对象,来测试 wait 和 sleep 方法,再通过执行结果的先后顺序证明:wait 方法会释放锁,而 sleep 方法并不会。同时我们还讲了几个 wait 和 sleep 的常见面试问题,希望本文可以帮助到你。

© 版权声明

相关文章