1. wait()和sleep()的区别
wait() 和 sleep() 有相同的地方,也有不同的地方。
【相同点】:都是让线程暂停一段时间的方法。
【不同点】:
(1)wait()是Object类的一个方法,sleep()是Thread类的一个方法。
(2)wait()必须放在synchronized修饰的代码块或方法中使用,sleep()可以放在任意位置使用。
(3)wait()被调用后,调用的wait()的线程进入BLOCKED状态,且会释放当前锁,可通过notify()或notifyAll()方法唤醒正在wait的线程。而sleep()被调用后当前线程进入TIMED_WAITING状态,不涉及到锁相关的操作。
2. 解释补充
对于不同点(3),调用wait()操作时,要执行三步操作:
<1> 释放当前锁
<2>让当前线程进入阻塞
<3>线程被唤醒时,重新获取到锁
因此,要使用wait()方法,只能在synchronized修饰的代码块或方法中使用,且调用wait()的线程进入阻塞后,需要等待其他线程调用notify()来唤醒。
代码如下:
public class Demo {
public static Object object = new Object();
public static void main(String[] args) throws InterruptedException {
Thread t1 = new Thread(() -> {
synchronized (object){
System.out.println("t1开始wait,进入阻塞!");
try {
object.wait();
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
System.out.println("t1被成功唤醒!");
}
});
Thread t2 = new Thread(() -> {
synchronized (object){
System.out.println("唤醒t1!");
object.notify();
}
});
t1.start();
t2.start();
}
}
如果调用了wait()的线程进入阻塞后,没有其他线程唤醒的话,这个线程会持续阻塞下去,直至有其他线程来唤醒。 代码如下
public class Demo {
public static Object object = new Object();
public static void main(String[] args) throws InterruptedException {
synchronized (object){
System.out.println("开始wait");
object.wait();
System.out.println("wait结束!");
}
}
}
结果是线程持续处于阻塞状态,一直无法结束,等待其他线程调用notify来唤醒。