在JAVA的学习中,不少人会把
sleep
和wait
搞混,认为都是做线程的等待,下面主要介绍下这俩者是什么,及了解它们之间的差异和相似之处。
一般差异
简单来说,wait()
是一个用于线程同步
的实例方法。它可以在任何对象上调用,因为它定义在java.lang.Object
上, 但只能从同步块中调用
。它释放对象上的锁,以便另一个线程可以跳入并获取锁。
另一方面,Thread.sleep()
是可以从任何上下文调用的静态方法。Thread.sleep()暂停当前线程
,不释放任何锁。
主要区别
sleep()
线程控制自身流程。wait()
用来线程间通信,使拥有该对象锁的线程等待直到指定时间或notify()。wait()
会释放锁和监视器,sleep()
不释放任何锁或监视器等。wait()
用于线程间通信,而sleep()
用于在执行时引入暂停- 适用区域,
wait
只能放在同步语句
块中才有意义。
注意事项
wait可以代替sleep吗?
不可以,如果直接调用wait
会抛出java.lang.IllegalMonitorStateException
异常,原因是还没有得到对象锁
,所以无法释放锁。
如何获取锁?
- 执行对象的
synchronized
实例方法。 - 执行对象对应类的
synchronized
静态方法。 - 执行对该对象加
synchronized
的同步代码块。
演示
Thread.sleep
1 | static void method1() { |
运行此示例一共将一共耗时10S
,因T1线程
获得锁后会暂停执行,这时候LOCK
依旧在T1线程
中,然后才会被T2线程
获取
1 | [T1] begin sleep ... |
Object.wait
1 | static void method2() { |
运行此示例会立即输出T1/T2 begin wait ...
但永远不会输出T1/T2 end wait ...
,因为没有线程调用LOCK.notify/notifyAll
将它们唤醒
1 | [T1] begin wait ... |
结论
调用:
1 | wait(): 对象调用,当前线程必须在持有锁的对象上进行同步。 |
同步:
1 | wait(): 同步多个线程同时访问一个对象时 |
锁:
1 | wait(): 释放锁,其它线程有机会执行 |
唤醒条件:
1 | wait(): 直到调用对象的 notify、notifyAll |
使用:
sleep(): 用于时间同步
wait(): 用于多线程同步
- 说点什么
- 个人QQ:1837307557
- battcn开源群(适合新手):391619659
微信公众号:battcn
(欢迎调戏)