一個小夥伴私信了一個小米的面試題,問題是: “執行緒池如何知道一個執行緒的任務已經執行完成”?
說實話,這個問題確實很刁鑽,畢竟像很多工作5年多的小夥伴,連執行緒池都沒用過,怎麼可能回答出來這個問題呢?
下面我們來看看普通人和高手遇到這個問題的回答思路。
普通人:
嗯..
高手:
好的,我會從兩個方面來回答。
-
線上程池內部,當我們把一個任務丟給執行緒池去執行,執行緒池會排程工作執行緒來執行這個任務的run方法,run方法正常結束,也就意味著任務完成了。
所以執行緒池中的工作執行緒是通過同步呼叫任務的
run()
方法並且等待run
方法返回後,再去統計任務的完成數量。 -
如果想線上程池外部去獲得執行緒池內部任務的執行狀態,有幾種方法可以實現。
-
執行緒池提供了一個
isTerminated()
方法,可以判斷執行緒池的執行狀態,我們可以迴圈判斷isTerminated()
方法的返回結果來了解執行緒池的執行狀態,一旦執行緒池的執行狀態是Terminated
,意味著執行緒池中的所有任務都已經執行完了。想要通過這個方法獲取狀態的前提是,程式中主動呼叫了執行緒池的shutdown()
方法。在實際業務中,一般不會主動去關閉執行緒池,因此這個方法在實用性和靈活性方面都不是很好。 -
線上程池中,有一個
submit()
方法,它提供了一個Future的返回值,我們通過Future.get()
方法來獲得任務的執行結果,當執行緒池中的任務沒執行完之前,future.get()
方法會一直阻塞,直到任務執行結束。因此,只要future.get()
方法正常返回,也就意味著傳入到執行緒池中的任務已經執行完成了! -
可以引入一個CountDownLatch計數器,它可以通過初始化指定一個計數器進行倒數計時,其中有兩個方法分別是
await()
阻塞執行緒,以及countDown()
進行倒數計時,一旦倒數計時歸零,所以被阻塞在await()
方法的執行緒都會被釋放。基於這樣的原理,我們可以定義一個CountDownLatch物件並且計數器為1,接著線上程池程式碼塊後面呼叫
await()
方法阻塞主執行緒,然後,當傳入到執行緒池中的任務執行完成後,呼叫countDown()
方法表示任務執行結束。最後,計數器歸零0,喚醒阻塞在
await()
方法的執行緒。
-
基於這個問題,我簡單總結一下,不管是執行緒池內部還是外部,要想知道執行緒是否執行結束,我們必須要獲取執行緒執行結束後的狀態,而執行緒本身沒有返回值,所以只能通過阻塞-喚醒的方式來實現,future.get和CountDownLatch都是這樣一個原理。
以上就是我對於這個問題的回答!
總結
大家可以站在面試官的角度來看高手的回答,
不難發現,高手對於技術基礎的掌握程度,是非常深和全面的。這也是面試官考察這類問題的目的。
因此,Mic提醒大家,除了日常的CRUD以外,抽出部分時間去做技術深度和廣度的學習是非常有必要的。
好的,本期的普通人VS高手面試系列就到這裡結束了,喜歡的朋友記得點贊收藏。
我是Mic,一個工作了14年的Java程式設計師,我們們下篇文章再見。