wait
wait方法来着Object的方法也就说所有对象都有这个方法,和它配套使用的有notify方法和notifyAll方法使用,但是这3个方法都要配合synchronized使如果直接使用会抛出IllegalMonitorStateException异常
public class Test1 {
public static void main(String[] args) throws InterruptedException {
final Object o = new Object();
new Thread(() -> {
synchronized (o) {
log.debug("开始睡眠");
try {
o.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
log.debug("你好");
}
},"t1").start();
new Thread(() -> {
synchronized (o) {
log.debug("起床了");
o.notifyAll();
}
},"t1").start();
TimeUnit.SECONDS.sleep(1);
}
}
在synchronized 代码块了代码成功执行。因为synchronized会把锁对象和一个Monitor对象关联
Monitor的Owner就是这个说的主人,一个synchronized同时只有一个线程执行,也对应了一个Monitor同时只有一个主人,EntryList就是同一个synchronized锁其他等待的线程,当前面的Owner执行完毕后EntryList里面的线程就会开始竞争谁成为Owner谁就开始执行,而WaitSet就是当线程抢到锁的时候他的角色是Owner,当他调用wait方法时,他会进入一个waitSet然后释放锁,当这个synchronized的其他线程调用notify方法和notifyAll时唤醒他,重新进去EntryList竞争
sleep
sleep 是线程类(Thread)的方法,导致此线程暂停执行指定时间,给执行机会给其他线程,但是监控状态依然保持,到时后会自动恢复。调用sleep 不会释放对象锁。
public class Test1 {
public static void main(String[] args) throws InterruptedException {
final Object o = new Object();
new Thread(() -> {
synchronized (o) {
log.debug("开始睡眠");
try {
TimeUnit.SECONDS.sleep(15);
} catch (InterruptedException e) {
e.printStackTrace();
}
log.debug("你好");
}
},"t1").start();
new Thread(() -> {
synchronized (o) {
log.debug("起床了");
}
},"t2").start();
TimeUnit.SECONDS.sleep(20);
}
}
我们发现在t1线程睡眠的时候t2线程是完全没办法干活的硬生生的等待了15秒才开始执行
总结
1.wait必须放在同步控制方法或者同步语句块中使用,而sleep方法则可以放在任何地方使用。
2.wait会释放同步锁,让其他线程进入synchronized代码块执行。sleep不会释放锁,其他线程只能等待在synchronized代码块中进入sleep的线程醒后执行完毕才能竞争持有锁。