多线程中wait()和sleep()以及notify()

多线程中会使用到两个延迟的函数,wait()和sleep()。一个唤醒函数notify().

  1. wait和notify是Object类中的方法,而sleep是Thread类中的方法。

  2. sleep是Thread类中的静态方法。无论是在a线程中调用b的sleep方法,还是b线程中调用a的sleep方法,谁调用,谁睡觉。最主要的是sleep方法调用之后,并没有释放锁,sleep是帮助其他线程获得运行机会的最好方法,但是如果当前线程获取到的有锁,sleep不会让出锁。使得线程仍然可以同步控制。sleep不会让出系统资源;而wait是进入线程等待池中等待,让出系统资源,释放对象锁。

  3. 调用wait方法的线程,不会自己唤醒,需要线程调用 notify / notifyAll 方法唤醒等待池中的所有线程,才会进入就绪队列中等待系统分配资源。sleep方法会自动唤醒,如果时间不到,想要唤醒,可以使用interrupt方法强行打断。
    Thread.sleep(0) // 触发操作系统立刻重新进行一次CPU竞争。

  4. 使用范围:sleep可以在任何地方使用。而wait,notify,notifyAll只能在同步控制方法或者同步控制块中使用。

  5. sleep必须捕获异常,而wait,notify,notifyAll的不需要捕获异常。若调用wait()没有持有适当的锁,执行wait后,会抛出异常IllegalMonitorException,这是个runtimeException的子类,不需要try/catch。

  6. 线程sleep()睡眠到期自动苏醒,并返回到可运行状态(就绪),不是运行状态。

  7. sleep()是静态方法,只能控制当前正在运行的线程.

notify():

notify方法执行后,只有等notify方法的线程将程序执行完,也就是退出synchronized代码块或者方法后,当前线程才会释放锁,呈wait状态的线程才会获得该对象的锁,当wait线程执行完毕后,释放该线程对象的锁,此时若该对象没有再次使用notify语句,即便该对象已经空闲,其他wait状态等待的线程也没有得到该对象的通知,还是会继续阻塞在wait状态,直到这个对象发出notify或者notifyAll。

notify 和 notifyAll的区别:

notify方法只唤醒一个等待(对象的)线程并使该线程开始执行。所以如果有多个线程等待一个对象,这个方法只会唤醒其中一个线程,选择哪个线程取决于操作系统对多线程管理的实现。notifyAll 会唤醒所有等待(对象的)线程,尽管哪一个线程将会第一个处理取决于操作系统的实现。如果当前情况下有多个线程需要被唤醒,推荐使用notifyAll 方法。比如在生产者-消费者里面的使用,每次都需要唤醒所有的消费者或是生产者,以判断程序是否可以继续往下执行。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章