【面試普通人VS高手系列】線程池如何知道一個線程的任務已經執行完成


一個小伙伴私信了一個小米的面試題,問題是: “線程池如何知道一個線程的任務已經執行完成”?

說實話,這個問題確實很***鑽,畢竟像很多工作5年多的小伙伴,連線程池都沒用過,怎么可能回答出來這個問題呢?

下面我們來看看普通人和高手遇到這個問題的回答思路。

普通人:

嗯..

高手:

好的,我會從兩個方面來回答。

  1. 在線程池內部,當我們把一個任務丟給線程池去執行,線程池會調度工作線程來執行這個任務的run方法,run方法正常結束,也就意味着任務完成了。

    所以線程池中的工作線程是通過同步調用任務的run()方法並且等待run方法返回后,再去統計任務的完成數量。

  2. 如果想在線程池外部去獲得線程池內部任務的執行狀態,有幾種方法可以實現。

    1. 線程池提供了一個isTerminated()方法,可以判斷線程池的運行狀態,我們可以循環判斷isTerminated()方法的返回結果來了解線程池的運行狀態,一旦線程池的運行狀態是Terminated,意味着線程池中的所有任務都已經執行完了。想要通過這個方法獲取狀態的前提是,程序中主動調用了線程池的shutdown()方法。在實際業務中,一般不會主動去關閉線程池,因此這個方法在實用性和靈活性方面都不是很好。

    2. 在線程池中,有一個submit()方法,它提供了一個Future的返回值,我們通過Future.get()方法來獲得任務的執行結果,當線程池中的任務沒執行完之前,future.get()方法會一直阻塞,直到任務執行結束。因此,只要future.get()方法正常返回,也就意味着傳入到線程池中的任務已經執行完成了!

    3. 可以引入一個CountDownLatch計數器,它可以通過初始化指定一個計數器進行倒計時,其中有兩個方法分別是await()阻塞線程,以及countDown()進行倒計時,一旦倒計時歸零,所以被阻塞在await()方法的線程都會被釋放。

      基於這樣的原理,我們可以定義一個CountDownLatch對象並且計數器為1,接着在線程池代碼塊后面調用await()方法阻塞主線程,然后,當傳入到線程池中的任務執行完成后,調用countDown()方法表示任務執行結束。

      最后,計數器歸零0,喚醒阻塞在await()方法的線程。

img

點擊並拖拽以移動

基於這個問題,我簡單總結一下,不管是線程池內部還是外部,要想知道線程是否執行結束,我們必須要獲取線程執行結束后的狀態,而線程本身沒有返回值,所以只能通過阻塞-喚醒的方式來實現,future.get和CountDownLatch都是這樣一個原理。

以上就是我對於這個問題的回答!

總結

大家可以站在面試官的角度來看高手的回答,

不難發現,高手對於技術基礎的掌握程度,是非常深和全面的。這也是面試官考察這類問題的目的。

因此,Mic提醒大家,除了日常的CRUD以外,抽出部分時間去做技術深度和廣度的學習是非常有必要的。

好的,本期的普通人VS高手面試系列就到這里結束了,喜歡的朋友記得點贊收藏。

我是Mic,一個工作了14年的Java程序員,咱們下篇文章再見。
img


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM