(InterrputedException e) {
等待的线程只是被激活,但是必须得再次获得锁才能继续往下执行,也就是说只要锁没被释放,原等待线程因为为获取锁仍然无法继续执行。
notify的作用只负责唤醒线程,线程被唤醒后有权利重新参与线程的调度。
wait()方法、notify()方法和notiftAll()方法用于协调多线程对共享数据的存取,所以只能在同步方法或者同步块中使用,否则抛出IllegalMonitorStateException。
两者的区别
(1)属于不同的两个类,sleep()方法是线程类(Thread)的静态方法,wait()方法是Object类里的方法。
(2)sleep()方法不会释放锁,wait()方法释放对象锁。
(3)sleep()方法可以在任何地方使用,wait()方法则只能在同步方法或同步块中使用。
(4)sleep()必须捕获异常,wait()方法、notify()方法和notiftAll()方法不需要捕获异常。
(5)sleep()使线程进入阻塞状态(线程睡眠),wait()方法使线程进入等待队列(线程挂起),也就是阻塞类别不同。
(6) 它们都可以被interrupted方法中断。
wait(1000)与sleep(1000)的区别
Thread.Sleep(1000)
意思是在未来的1000毫秒内本线程不参与CPU竞争,1000毫秒过去之后,这时候也许另外一个线程正在使用CPU,那么这时候操作系统是不会重新分配CPU的,直到那个线程挂起或结束。
即使这个时候恰巧轮到操作系统进行CPU 分配,那么当前线程也不一定就是总优先级最高的那个,CPU还是可能被其他线程抢占去。
另外值得一提的是Thread.Sleep(0)的作用,就是触发操作系统立刻重新进行一次CPU竞争,竞争的结果也许是当前线程仍然获得CPU控制权,也许会换成别的线程获得CPU控制权。
wait(1000)
表示将锁释放1000毫秒,到时间后如果锁没有被其他线程占用,则再次得到锁,然后wait方法结束,执行后面的代码,如果锁被其他线程占用,则等待其他线程释放锁。
注意,设置了超时时间的wait方法一旦过了超时时间,并不需要其他线程执行notify也能自动解除阻塞,但是如果没设置超时时间的wait方法必须等待其他线程执行notify。
public class test{
public static void main(String[] args) {
Integer i = new Integer(1);
new Thread(new waitThread(i)).start();
new Thread(new notifyThread(i)).start();
class waitThread implements Runnable{
Integer i;
public waitThread(Integer i){
super();
this.i = i;
@Override
public void run(){
try{
synchronized(i){
long start = System.currentTimeMillis();
i.wait(1000);
System.out.println("waitThread "+(System.currentTimeMillis()-start)+" done");
}catch(InterruptedException e){
e.printStackTrace();
class notifyThread implements Runnable{
Integer i;
public notifyThread(Integer i){
super();
this.i = i;
@Override
public void run(){
try{
long start = System.currentTimeMillis();
//如果此处设成1500,因为sleep没有占有锁,wait方法在1000ms后会自动再次获得锁然后解除阻塞执行。
Thread.sleep(500);
synchronized(i){
Thread.sleep(1500);
//如果wait过了超时时间,无论有无notify,wait都会自动解除阻塞,即该句可以注释,不影响结果。
//但是如果wait没有设置超时时间,该句必须存在,否则waitThread用于处于阻塞状态。
i.notify();
System.out.println("notifyThread "+(System.currentTimeMillis()-start)+" done");
}catch(InterruptedException e){
e.printStackTrace();
join()方法:
join()方法使调用该方法的线程在此之前执行完毕,也就是等待该方法的线程执行完毕后再往下继续执行。注意该方法也需要捕捉异常。
yield()方法:
该方法与sleep()类似,都是可以让当前正在运行的线程暂停,区别在于yield()方法不会阻塞该线程,它只是将线程转换成就绪状态,让系统的调度器重新调度一次,并且yield()方法只能让优先级相同或许更高的线程有执行的机会。
很想高飞,但我不能;不想天空,剩我一人。