【正确答案】
【答案解析】sleep()是使线程暂停执行一段时间的方法。wait()也是一种使线程暂停执行的方法,例如,当线程交互时,如果线程对一个同步对象X发出一个wait()调用请求,那么该线程会暂停执行,被调对象进入等待状态,直到被唤醒或等待时间超时。
具体而言,sleep()方法与wait()方法的区别主要表现在以下几个方面:
1)原理不同。sleep()方法是Thread类的静态方法,是线程用来控制自身流程的,它会使此线程暂停执行一段时间,而把执行机会让给其他线程,等到计时时间一到,此线程会自动“苏醒”,例如,当线程执行报时功能时,每一秒钟打印出一个时间,那么此时就需要在打印方法前面加上一个sleep()方法,以便让自己每隔1s执行一次,该过程如同闹钟一样。而wait()方法是Object类的方法,用于线程间的通信,这个方法会使当前拥有该对象锁的进程等待,直到其他线程调用notify()方法(或notifyAll方法)时才“醒”来,不过开发人员也可以给它指定一个时间,自动“醒”来。与wait()方法配套的方法还有notify()方法和notifyAlI()方法。
2)对锁的处理机制不同。由于sleep()方法的主要作用是让线程暂停执行一段时间,时间一到则自动恢复,不涉及线程间的通信,因此,调用sleep()方法并不会释放锁。而wait()方法则不同,当调用wait()方法后,线程会释放掉它所占用的锁,从而使线程所在对象中的其他synchronized数据可被别的线程使用。举个简单例子,如果何吴拿遥控器的期间,可以用自己的sleep()方法每隔10min调一次频道,而在这10min里,遥控器还在他的手上。
3)使用区域不同。由于wait()方法的特殊意义,因此它必须放在同步控制方法或者同步语句块中使用,而sleep()方法则可以放在任何地方使用。
sleep()方法必须捕获异常,而wait()、notify()以及notifyall()不需要捕获异常。在sleep的过程中,有可能被其他对象调用它的interrupt(),产生InterruptedException异常。
由于sleep不会释放“锁标志”,容易导致死锁问题的发生,因此,一般情况下,不推荐使用sleep()方法,而推荐使用wait()方法。
引申:sleep()方法与yield()方法有什么区别?
sleep()方法与yield()方法的区别主要表现在以下几个方面:
1)sleep()方法给其他线程运行机会时不考虑线程的优先级,因此会给低优先级的线程以运行的机会,而yield()方法只会给相同优先级或更高优先级的线程以运行的机会。
2)线程执行sleep()方法后会转入阻塞状态,所以,执行sleep()方法的线程在指定的时间内肯定不会被执行,而yield()方法只是使当前线程重新回到可执行状态,所以执行yield()方法的线程有可能在进入到可执行状态后马上又被执行。
3)sleep()方法声明抛出InterruptedException,而yield()方法没有声明任何异常。
4)sleep()方法比yield()方法(跟操作系统相关)具有更好的可移植性。
常见笔试题:
1.利用Thread.wait()同步线程,可以设置超时时间吗?
A.可以 B. 不可以
答案:A。可以设置超时,函数原型为wait(long timeout)和wait(long timeout,int nanos)timeout代表最长的等待时间,单位为ms;nanos代表额外的等待时间,单位为ns。
2.在一个线程中sleep(1000)方法,将使该线程在多长时间后获得对CPU的控制(假设睡眠过程中不会有其他事件唤醒该线程)?
A.正好1000ms B.少于1000ms C.大于等于1000ms D.不一定
答案:C。sleep()方法指定的时间为线程不会运行的最短时间。当睡眠时间结束后,线程会返回到可运行状态,不是运行状态,还需要等待CPU调度执行。因此,sleep()方法不能保证该线程睡眠到期后就开始执行。