[[譯]Java中Wait、Sleep和Yield方法的區別](http://www.jianshu.com/p/25e959037eed)
原文地址:[Difference between Wait and Sleep, Yield in Java](http://javarevisited.blogspot.com/2011/12/difference-between-wait-sleep-yield.html)
Java中wait、sleep的區別或者Java中sleep、yield的區別是Java面試或者多線程面試中最常問的問題之一。在這3個在Java中能夠用來暫停線程的方法中,sleep()和yield()方法是定義在Thread類中,而wait()方法是定義在Object類中的, 這也是面試中常問的一個問題。
wait()和sleep()的關鍵的區別在于,wait()是用于線程間通信的,而sleep()是用于短時間暫停當前線程。更加明顯的一個區別在于,當一個線程調用wait()方法的時候,會釋放它鎖持有的對象的管程和鎖,但是調用sleep()方法的時候,不會釋放他所持有的管程。
回到yield()方法上來,與wait()和sleep()方法有一些區別,它僅僅釋放線程所占有的CPU資源,從而讓其他線程有機會運行,但是并不能保證某個特定的線程能夠獲得CPU資源。誰能獲得CPU完全取決于調度器,在有些情況下調用yield方法的線程甚至會再次得到CPU資源。所以,依賴于yield方法是不可靠的,它只能盡力而為。
## Wait vs Sleep vs Yield in Java
### Java中wait和sleep的區別
wait和sleep的主要區別是調用wait方法時,線程在等待的時候會釋放掉它所獲得的monitor,但是調用Thread.sleep()方法時,線程在等待的時候仍然會持有monitor或者鎖。另外,Java中的wait方法應在同步代碼塊中調用,但是sleep方法不需要。
另一個區別是Thread.sleep()方法是一個靜態方法,作用在當前線程上;但是wait方法是一個實例方法,并且只能在其他線程調用本實例的notify()方法時被喚醒。另外,使用sleep方法時,被暫停的線程在被喚醒之后會立即進入就緒態(Runnable state),但是使用wait方法的時候,被暫停的線程會首先獲得鎖(譯者注:阻塞態),然后再進入就緒態。所以,根據你的需求,如果你需要暫定你的線程一段特定的時間就使用sleep()方法,如果你想要實現線程間通信就使用wait()方法。
下面列出Java中wait和sleep方法的區別:
1. wait只能在同步(synchronize)環境中被調用,而sleep不需要。詳見[Why to wait and notify needs to call from synchronized method](http://javarevisited.blogspot.com/2011/05/wait-notify-and-notifyall-in-java.html)
2. 進入wait狀態的線程能夠被notify和notifyAll線程喚醒,但是進入sleeping狀態的線程不能被notify方法喚醒。
3. wait通常有條件地執行,線程會一直處于wait狀態,直到某個條件變為真。但是sleep僅僅讓你的線程進入睡眠狀態。
4. wait方法在進入wait狀態的時候會釋放對象的鎖,但是sleep方法不會。
5. wait方法是針對一個被同步代碼塊加鎖的對象,而sleep是針對一個線程。更詳細的講解可以參考《Java核心技術卷1》,里面介紹了如何使用wait和notify方法。
### yield和sleep的區別
yield和sleep的主要是,yield方法會臨時暫停當前正在執行的線程,來讓有同樣優先級的正在等待的線程有機會執行。如果沒有正在等待的線程,或者所有正在等待的線程的優先級都比較低,那么該線程會繼續運行。執行了yield方法的線程什么時候會繼續運行由線程調度器來決定,不同的廠商可能有不同的行為。yield方法不保證當前的線程會暫停或者停止,但是可以保證當前線程在調用yield方法時會放棄CPU。
在Java中Sleep方法有兩個, 一個只有一個毫秒參數,另一個有毫秒和納秒兩個參數。
~~~
sleep(long millis)
~~~
or
~~~
sleep(long millis, int nanos)
~~~
會讓當前執行的線程sleep指定的時間。
下面這張圖很好地展示了在調用wait、sleep、yield方法的時候,線程狀態如何轉換。

Paste_Image.png
Java中sleep方法的幾個注意點:
1. Thread.sleep()方法用來暫停線程的執行,將CPU放給線程調度器。
2. Thread.sleep()方法是一個靜態方法,它暫停的是當前執行的線程。
3. Java有兩種sleep方法,一個只有一個毫秒參數,另一個有毫秒和納秒兩個參數。
4. 與wait方法不同,sleep方法不會釋放鎖
5. 如果其他的線程中斷了一個休眠的線程,sleep方法會拋出Interrupted Exception。
6. 休眠的線程在喚醒之后不保證能獲取到CPU,它會先進入就緒態,與其他線程競爭CPU。
7. 有一個易錯的地方,當調用t.sleep()的時候,會暫停線程t。這是不對的,因為Thread.sleep是一個靜態方法,它會使當前線程而不是線程t進入休眠狀態。
這就是java中的sleep方法。我們已經看到了java中sleep、wait以及yield方法的區別。總之,記住sleep和yield作用于當前線程。
- JVM
- 深入理解Java內存模型
- 深入理解Java內存模型(一)——基礎
- 深入理解Java內存模型(二)——重排序
- 深入理解Java內存模型(三)——順序一致性
- 深入理解Java內存模型(四)——volatile
- 深入理解Java內存模型(五)——鎖
- 深入理解Java內存模型(六)——final
- 深入理解Java內存模型(七)——總結
- Java內存模型
- Java內存模型2
- 堆內內存還是堆外內存?
- JVM內存配置詳解
- Java內存分配全面淺析
- 深入Java核心 Java內存分配原理精講
- jvm常量池
- JVM調優總結
- JVM調優總結(一)-- 一些概念
- JVM調優總結(二)-一些概念
- VM調優總結(三)-基本垃圾回收算法
- JVM調優總結(四)-垃圾回收面臨的問題
- JVM調優總結(五)-分代垃圾回收詳述1
- JVM調優總結(六)-分代垃圾回收詳述2
- JVM調優總結(七)-典型配置舉例1
- JVM調優總結(八)-典型配置舉例2
- JVM調優總結(九)-新一代的垃圾回收算法
- JVM調優總結(十)-調優方法
- 基礎
- Java 征途:行者的地圖
- Java程序員應該知道的10個面向對象理論
- Java泛型總結
- 序列化與反序列化
- 通過反編譯深入理解Java String及intern
- android 加固防止反編譯-重新打包
- volatile
- 正確使用 Volatile 變量
- 異常
- 深入理解java異常處理機制
- Java異常處理的10個最佳實踐
- Java異常處理手冊和最佳實踐
- Java提高篇——對象克隆(復制)
- Java中如何克隆集合——ArrayList和HashSet深拷貝
- Java中hashCode的作用
- Java提高篇之hashCode
- 常見正則表達式
- 類
- 理解java類加載器以及ClassLoader類
- 深入探討 Java 類加載器
- 類加載器的工作原理
- java反射
- 集合
- HashMap的工作原理
- ConcurrentHashMap之實現細節
- java.util.concurrent 之ConcurrentHashMap 源碼分析
- HashMap的實現原理和底層數據結構
- 線程
- 關于Java并發編程的總結和思考
- 40個Java多線程問題總結
- Java中的多線程你只要看這一篇就夠了
- Java多線程干貨系列(1):Java多線程基礎
- Java非阻塞算法簡介
- Java并發的四種風味:Thread、Executor、ForkJoin和Actor
- Java中不同的并發實現的性能比較
- JAVA CAS原理深度分析
- 多個線程之間共享數據的方式
- Java并發編程
- Java并發編程(1):可重入內置鎖
- Java并發編程(2):線程中斷(含代碼)
- Java并發編程(3):線程掛起、恢復與終止的正確方法(含代碼)
- Java并發編程(4):守護線程與線程阻塞的四種情況
- Java并發編程(5):volatile變量修飾符—意料之外的問題(含代碼)
- Java并發編程(6):Runnable和Thread實現多線程的區別(含代碼)
- Java并發編程(7):使用synchronized獲取互斥鎖的幾點說明
- Java并發編程(8):多線程環境中安全使用集合API(含代碼)
- Java并發編程(9):死鎖(含代碼)
- Java并發編程(10):使用wait/notify/notifyAll實現線程間通信的幾點重要說明
- java并發編程-II
- Java多線程基礎:進程和線程之由來
- Java并發編程:如何創建線程?
- Java并發編程:Thread類的使用
- Java并發編程:synchronized
- Java并發編程:Lock
- Java并發編程:volatile關鍵字解析
- Java并發編程:深入剖析ThreadLocal
- Java并發編程:CountDownLatch、CyclicBarrier和Semaphore
- Java并發編程:線程間協作的兩種方式:wait、notify、notifyAll和Condition
- Synchronized與Lock
- JVM底層又是如何實現synchronized的
- Java synchronized詳解
- synchronized 與 Lock 的那點事
- 深入研究 Java Synchronize 和 Lock 的區別與用法
- JAVA編程中的鎖機制詳解
- Java中的鎖
- TreadLocal
- 深入JDK源碼之ThreadLocal類
- 聊一聊ThreadLocal
- ThreadLocal
- ThreadLocal的內存泄露
- 多線程設計模式
- Java多線程編程中Future模式的詳解
- 原子操作(CAS)
- [譯]Java中Wait、Sleep和Yield方法的區別
- 線程池
- 如何合理地估算線程池大小?
- JAVA線程池中隊列與池大小的關系
- Java四種線程池的使用
- 深入理解Java之線程池
- java并發編程III
- Java 8并發工具包漫游指南
- 聊聊并發
- 聊聊并發(一)——深入分析Volatile的實現原理
- 聊聊并發(二)——Java SE1.6中的Synchronized
- 文件
- 網絡
- index
- 內存文章索引
- 基礎文章索引
- 線程文章索引
- 網絡文章索引
- IOC
- 設計模式文章索引
- 面試
- Java常量池詳解之一道比較蛋疼的面試題
- 近5年133個Java面試問題列表
- Java工程師成神之路
- Java字符串問題Top10
- 設計模式
- Java:單例模式的七種寫法
- Java 利用枚舉實現單例模式
- 常用jar
- HttpClient和HtmlUnit的比較總結
- IO
- NIO
- NIO入門
- 注解
- Java Annotation認知(包括框架圖、詳細介紹、示例說明)