1.概览
在这篇简单的文章中,我们将会看到标准的sleep()和wait()方法,并且理解他们的区别和相似性。
2.wait和sleep的一般区别
简单点说,wait()方法是个实例方法,它主要用于线程同步。它可以被任何对象调用,因为这个方法是被定义在Object类里的。但是它只能从一个同步代码块中被调用。只有它释放了对象上的锁后,其他线程才能进入并获取锁。
另一方面,Thread.sleep()是一个静态方法,它可以在任何环境中被调用。Thread.sleep() 会暂停当前线程并且不会释放任何锁。
这里有一个关于这俩个API的例子:
privatestaticObject LOCK = newObject();
privatestaticvoidsleepWaitExamples()
throwsInterruptedException {
Thread.sleep(1000);
System.out.println( "Thread '"+ Thread.currentThread().getName() + "' is woken after sleeping for 1 second");
synchronized(LOCK) {
LOCK.wait(1000);
System.out.println("Object '"+ LOCK + "' is woken after"+
" waiting for 1 second");
}
}
3.唤醒 Wait和Sleep
当我们使用sleep()方法时,线程会在指定的时间间隔内启动(get started),除非他被中断了。
对于wait()方法来说,唤醒处理要更复杂一些。调用当前正在等待的监视器的notify()或 notifyAll()方法可以唤醒该线程。
你可以使用notifyAll()方法唤醒所有处于等待状态下的线程。和wait()方法类似,notify()和notifyAll()方法只能在一个同步上下文中被调用。
例如,这里有一个wait的例子:
synchronized(b) {
while(b.sum == 0) {
System.out.println("Waiting for ThreadB to complete...");
b.wait();
}
System.out.println("ThreadB has completed. "+
"Sum from that thread is: "+ b.sum);
}
之后,下面会演示另一个线程是如何会唤醒该等待的线程----通过调用monitor对象上的notify()方法:
int sum;
@Override
public void run() {
synchronized (this) {
int i = 0;
while (i < 100000) {
sum += i;
i++;
}
notify();
}
}
运行这个例子将会产生下面的输出:
Waiting for ThreadB to complete…
ThreadB has completed. Sum from that thread is: 704982704
4.总结
这篇文章是对wait和sleep()语法的一个入门。
通常来说,我们可以使用sleep()去控制一个线程的执行时间,使用wait()方法去控制多线程同步,当然了,在理解完这些最基本的概念之后,还有好多东西需要我们去探索。 和往常一样,你可以从这个地址处https://github.com/eugenp/tutorials/tree/master/core-java-concurrency/src/main/java/com/baeldung/concurrent/sleepwait去check案例代码。