面试题10解析
题目:分别说一下notify()/notifyAll()/sleep()/wait()的区别,以及如何实现多线程之间的通讯和协作?
本文阅读大概需要12分钟。
这个题目主要考查多线程的等待问题,以及它们之间的通讯和协作。
notify()/notifyAll()/sleep()/wait()的区别
notify():随机唤醒一个等待该对象同步锁的线程,进入就绪队列等待CPU的调度;这里的唤醒是由JVM确定唤醒哪个线程,而且不是按优先级决定。
notifyAll():唤醒所有的等待该对象同步锁的线程,进入就绪队列等待CPU调度;注意唤醒的是notify之前wait的线程,对于notify之后的wait线程是没有效果的。
wait():调用时需要先获得该Object的锁,调用后,会把当前的锁释放掉同时阻塞住;但可以通过调用该Object的notify()或者notifyAll()来重新获得锁。
sleep():在指定的时间内让正在执行的线程暂停执行,但不会释放锁。
区别主要体现在这几个方面?
我们通过对这些方法分析,sleep()方法属于Thread类,而wait()/notify()/notifyAll()属于Object基础类,也就是说每个对象都有wait()/notify()/notifyAll()的功能。
sleep()不会释放锁,而wait()会释放锁。
sleep()必须捕获异常,而wait()/notify()/notifyAll()不需要捕获异常。
sleep()可以在任何地方使用,而wait()/notify()/notifyAll()只能在同步控制方法或者同步控制块里面使用。
如何实现多线程之间的通讯和协作
利用同步和互斥来解决多线程之间的通讯和协作;可以说资源互斥、协调竞争是要解决的因,而同步是竞争协调的果。
通过synchronized/notify/notifyAll来实现线程之间的通信。
利用了Java5中提供的Lock/Condition来实现线程之间的相互通信。
使用信号量,如:CyclicBarrier/Semaphore/Countdownbatch。
怎么解决多线程计算的结果统计?
可以用join()以及Future/FutureTask来解决。join()的功能是使异步执行的线程变成同步执行;使用join()后,直到这个线程退出,程序才会往下执行。
记住,这里全部都是干货!!!