在JAVA的学习中,不少人会把
sleep
和wait
搞混,认为都是做线程的等待,下面主要介绍下这俩者是什么,及了解它们之间的差异和相似之处。
<!-- more -->
一般差异
简单来说,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
static void method1() {
synchronized (LOCK) {
try {
System.out.println("[" + Thread.currentThread().getName() + "] begin sleep ...");
Thread.sleep(5_000);
System.out.println("[" + Thread.currentThread().getName() + "] end sleep ...");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) {
Stream.of("T1", "T2").forEach(name -> new Thread(DifferenceOfWaitAndSleep::method1, name).start());
}
运行此示例一共将一共耗时10S
,因T1线程
获得锁后会暂停执行,这时候LOCK
依旧在T1线程
中,然后才会被T2线程
获取
[T1] begin sleep ...
[T1] end sleep ...
[T2] begin sleep ...
[T2] end sleep ...
Object.wait
static void method2() {
synchronized (LOCK) {
try {
System.out.println("[" + Thread.currentThread().getName() + "] begin wait ...");
LOCK.wait();
System.out.println("[" + Thread.currentThread().getName() + "] end wait ...");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) {
Stream.of("T1", "T2").forEach(name -> new Thread(DifferenceOfWaitAndSleep::method2, name).start());
}
运行此示例会立即输出T1/T2 begin wait ...
但永远不会输出T1/T2 end wait ...
,因为没有线程调用LOCK.notify/notifyAll
将它们唤醒
[T1] begin wait ...
[T2] begin wait ...
结论
调用:
wait(): 对象调用,当前线程必须在持有锁的对象上进行同步。
sleep(): 线程调用,始终执行的当前线程
同步:
wait(): 同步多个线程同时访问一个对象时
sleep(): 多个线程同步等待线程休眠
锁:
wait(): 释放锁,其它线程有机会执行
sleep(): 持有锁,固定时间后释放或被打断(interrupt)
唤醒条件:
wait(): 直到调用对象的 notify、notifyAll
sleep(): 直到至少时间到期或者调用interrupt()
使用:
sleep(): 用于时间同步
wait(): 用于多线程同步
- 说点什么
- 个人QQ:1837307557
- battcn开源群(适合新手):391619659
微信公众号:battcn
(欢迎调戏)
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。