成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

追蹤解析 FutureTask 源碼

xcc3641 / 571人閱讀

摘要:零前期準(zhǔn)備文章異常啰嗦且繞彎。版本版本簡介是中默認(rèn)的實(shí)現(xiàn)類,常與結(jié)合進(jìn)行多線程并發(fā)操作。所以方法的主體其實(shí)就是去喚醒被阻塞的線程。本文僅為個(gè)人的學(xué)習(xí)筆記,可能存在錯(cuò)誤或者表述不清的地方,有緣補(bǔ)充

零 前期準(zhǔn)備 0 FBI WARNING

文章異常啰嗦且繞彎。

1 版本

JDK 版本 : OpenJDK 11.0.1

IDE : idea 2018.3

2 ThreadLocal 簡介

FutureTask 是 jdk 中默認(rèn)的 Future 實(shí)現(xiàn)類,常與 Callable 結(jié)合進(jìn)行多線程并發(fā)操作。

3 Demo
import java.util.concurrent.*;

public class FutureTaskDemo {

    public static void main(String[] args) throws ExecutionException, InterruptedException, TimeoutException {
        
        //創(chuàng)建一個(gè)線程池
        ExecutorService pool = Executors.newFixedThreadPool(1);
        try{
            //創(chuàng)建一個(gè)要執(zhí)行的 Callable 對象
            //此處其實(shí) Runnable 對象也可以,但是通常不會(huì)那樣做
            Callable task = () -> {
                //休眠三秒
                TimeUnit.SECONDS.sleep(3);
                //返回一個(gè)字符串
                return "hello";
            };

            //用 FutureTask 對象去包裝 Callable
            FutureTask futureTask = new FutureTask<>(task);

            //此處將 FutureTask 對象丟進(jìn)線程池里
            pool.submit(futureTask);

            //注意,此處的 futureTask 本質(zhì)上是作為 Runnable 被丟進(jìn)池子里的
            //所以也可以用線程池的 execute(...) 方法
            //pool.execute(futureTask)

            //還有一種更常見的執(zhí)行方式是直接使用 Thread
            //new Thread(futureTask).start();

            //獲取結(jié)果
            //注意,如果沒有獲取到的話此處會(huì)阻塞線程直到獲取到為止
            String result = futureTask.get();

            //還有一種限時(shí)策略的結(jié)果獲取
            //超時(shí)的情況下會(huì)拋出異常
            //String result = futureTask.get(1,TimeUnit.SECONDS);

            System.out.println(result);
        }finally {
            //關(guān)閉連接池
            pool.shutdown();
        }

    }
}
一 FutureTask 的創(chuàng)建

回到 Demo 中的創(chuàng)建代碼:

FutureTask futureTask = new FutureTask<>(task);

追蹤 FutureTask 的構(gòu)造器:

//FutureTask.class
public FutureTask(Callable callable) {
    //有效性判斷,不能為空
    if (callable == null)
        throw new NullPointerException();
    //記錄下 callable 對象
    this.callable = callable;
    //state 是一個(gè) int 類型的對象,是一個(gè)
    //NEW = 0
    this.state = NEW;
}
二 run

FutureTask 本身是 Runnable 的實(shí)現(xiàn)類,其在被 ThreadPoolExecutor 或者 Thread 對象消費(fèi)的時(shí)候也是被當(dāng)做 Runnable 的實(shí)現(xiàn)類的。

所以其本身的核心邏輯就必然在 run() 方法中:

//FutureTask.class
public void run() {

    //先判斷狀態(tài),如果狀態(tài)不是 NEW 就會(huì)直接返回
    //RUNNER 是一個(gè) VarHandler 類型的變量,指向了 FutureTask 中的 thread 變量,用于儲(chǔ)存當(dāng)前的線程
    //但是如果 thread 已經(jīng)不為 null,此處也會(huì)直接返回
    //這兩種返回條件都意味著此 FutureTask 的 run() 方法已經(jīng)執(zhí)行過了
    if (state != NEW || !RUNNER.compareAndSet(this, null, Thread.currentThread()))
        return;

    try {
        //獲取 callable
        Callable c = callable;
        if (c != null && state == NEW) {
            V result;
            boolean ran;
            try {
                //執(zhí)行 callable 的業(yè)務(wù)邏輯
                result = c.call();
                //ran 為成功標(biāo)識(shí)
                ran = true;
            } catch (Throwable ex) {
                //出錯(cuò)的情況下
                result = null;
                ran = false;
                //不成功的情況下存入 exception
                setException(ex);
            }
            //如果成功的話會(huì)在此處進(jìn)行操作
            if (ran)
                set(result);
        }
    } finally {
        //置空
        runner = null;
        int s = state;
        if (s >= INTERRUPTING)
            //如果此 FutreTask 的狀態(tài)是中斷狀態(tài),會(huì)在此處不斷調(diào)用 Thread.yield() 空轉(zhuǎn)
            handlePossibleCancellationInterrupt(s);
    }
}

此處有兩個(gè)關(guān)鍵方法,即為 setException(...) 和 set(...):

//FutureTask.class
protected void setException(Throwable t) {
    //用 CAS 操作比較并更新狀態(tài)值
    if (STATE.compareAndSet(this, NEW, COMPLETING)) {
        //outcome 是一個(gè) Object 對象,用于存儲(chǔ) callable 的返回值
        //此處由于報(bào)錯(cuò)了,所以儲(chǔ)存的是錯(cuò)誤對象
        outcome = t;
        //EXCEPTIONAL = 3
        STATE.setRelease(this, EXCEPTIONAL);
        //最后清理工作,主要用于喚醒等待線程和執(zhí)行 callable
        finishCompletion();
    }
}

//FutureTask.class
protected void set(V v) {
    //基本邏輯和 setException(...) 方法雷同,只是 STATE 和 outcome 的儲(chǔ)存值不同
    if (STATE.compareAndSet(this, NEW, COMPLETING)) {
        outcome = v;
        STATE.setRelease(this, NORMAL);
        finishCompletion();
    }
}

再來看 finishCompletion() 方法:

//FutureTask.class
private void finishCompletion() {
    //WaitNode 是 FutureTask 的靜態(tài)內(nèi)部類
    //其本質(zhì)上是單向鏈表的節(jié)點(diǎn)表示類,用于存放想要獲取 Callable 的返回值但是被阻塞的線程的線程對象
    for (WaitNode q; (q = waiters) != null;) {
        //此處使用 CAS 將 q 從 WAITERS 里去除
        if (WAITERS.weakCompareAndSet(this, q, null)) {
            for (;;) {
                Thread t = q.thread;
                if (t != null) {
                    //此處置空線程對象,幫助 GC
                    q.thread = null;
                    //喚醒線程
                    LockSupport.unpark(t);
                }
                //接著往下遍歷
                WaitNode next = q.next;
                if (next == null)
                    break;
                q.next = null; 
                q = next;
            }
            break;
        }
    }
    //此方法是空的
    done();
    //置空 callable
    callable = null;
}

之前提到過在 FutureTask 的 get(...) 方法中會(huì)阻塞線程,直到 Callable 執(zhí)行完畢并能夠獲取返回值的時(shí)候才會(huì)結(jié)束阻塞。

所以 finishCompletion() 方法的主體其實(shí)就是去喚醒被阻塞的線程。

三 get

回到 Demo 中的創(chuàng)建代碼:

String result = futureTask.get();

追蹤 get() 方法:

//step 1
//FutureTask.class
public V get() throws InterruptedException, ExecutionException {
    //此處先判斷狀態(tài)值,如果非 COMPLETING,即為還沒完成,就會(huì)調(diào)用 awaitDone(...) 方法阻塞線程
    int s = state;
    if (s <= COMPLETING)
        s = awaitDone(false, 0L);
    //返回結(jié)果
    return report(s);
}

//step 2
//FutureTask.class
private V report(int s) throws ExecutionException {
    //獲取需要返回的對象
    Object x = outcome;
    //如果是正常結(jié)束的就直接返回對象即可
    if (s == NORMAL)
        return (V)x;
    //出錯(cuò)的情況下,拋異常
    if (s >= CANCELLED)
        throw new CancellationException();
    throw new ExecutionException((Throwable)x);
}

再來看一下阻塞線程的 awaitDone(...) 方法:

private int awaitDone(boolean timed, long nanos) throws InterruptedException {
    
    //循環(huán)的次數(shù)
    long startTime = 0L;
    //節(jié)點(diǎn)對象
    WaitNode q = null;
    //鏈表隊(duì)列標(biāo)識(shí),代表該線程是否被加入鏈表中,初始為 false 代表未被加入
    boolean queued = false;
    for (;;) {
        int s = state;
        if (s > COMPLETING) { //如果 Callable 的執(zhí)行已經(jīng)完成
            if (q != null)
                q.thread = null;
            return s;
        }else if (s == COMPLETING) //Callable 的執(zhí)行剛剛完成,后續(xù)工作還沒做
            Thread.yield();
        else if (Thread.interrupted()) {
            //線程被中斷了,會(huì)拋出錯(cuò)誤
            removeWaiter(q);
            throw new InterruptedException();
        } else if (q == null) { //進(jìn)入此處的判斷證明 Callable 還未完成,所以會(huì)創(chuàng)建等待節(jié)點(diǎn)
            //此處的 timed 傳入為 false,不會(huì)在此返回
            if (timed && nanos <= 0L)
                return s;
            q = new WaitNode(); //新建節(jié)點(diǎn)
        }else if (!queued)
            //queued 初始為 false,進(jìn)入此處的時(shí)候會(huì)將上一個(gè)判斷條件中新建的 q 加入到鏈表的首節(jié)點(diǎn)中
            //并且 queued 變成 true
            queued = WAITERS.weakCompareAndSet(this, q.next = waiters, q);
        else if (timed) {
            //如果此操作是限時(shí)的,那么這里需要判斷時(shí)間
            final long parkNanos;
            if (startTime == 0L) {
                startTime = System.nanoTime();
                if (startTime == 0L)
                    startTime = 1L;
                parkNanos = nanos;
            } else {
                long elapsed = System.nanoTime() - startTime;
                if (elapsed >= nanos) {
                    removeWaiter(q);
                    return state;
                }
                parkNanos = nanos - elapsed;
            }
            if (state < COMPLETING)
                //此處掛起線程,時(shí)間為 parkNanos
                //本例中傳入為 0L,所以是永久掛起
                LockSupport.parkNanos(this, parkNanos);
        }else
            //永久掛起線程
            LockSupport.park(this);
    }
}
四 一點(diǎn)嘮叨

FutureTask 和 ThreadLocal 一樣,都是 java.util.current 包中的小工具,封裝不復(fù)雜,理解即可。

本文僅為個(gè)人的學(xué)習(xí)筆記,可能存在錯(cuò)誤或者表述不清的地方,有緣補(bǔ)充

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/73421.html

相關(guān)文章

  • FutureTask源碼解析(1)——預(yù)備知識(shí)

    摘要:在分析它的源碼之前我們需要先了解一些預(yù)備知識(shí)。因?yàn)榻涌跊]有返回值所以為了與兼容我們額外傳入了一個(gè)參數(shù)使得返回的對象的方法直接執(zhí)行的方法然后返回傳入的參數(shù)。 前言 系列文章目錄 FutureTask 是一個(gè)同步工具類,它實(shí)現(xiàn)了Future語義,表示了一種抽象的可生成結(jié)果的計(jì)算。在包括線程池在內(nèi)的許多工具類中都會(huì)用到,弄懂它的實(shí)現(xiàn)將有利于我們更加深入地理解Java異步操作實(shí)現(xiàn)。 在分析...

    mmy123456 評論0 收藏0
  • 【Java并發(fā)】Runnable、Callable、Future、FutureTask

    摘要:聲明了幾種方法,其中有一個(gè)就是傳入聲明了對具體的或者任務(wù)執(zhí)行進(jìn)行取消查詢結(jié)果獲取等方法。事實(shí)上,是接口的一個(gè)唯一實(shí)現(xiàn)類。使用示例第一種方式是使用繼承了的線程池中的方法,將直接提交創(chuàng)建。 創(chuàng)建線程的兩種方式 直接繼承 Thread 實(shí)現(xiàn) Runnable 接口 這兩種方式都有一個(gè)缺點(diǎn):在執(zhí)行完成任務(wù)之后,無法直接獲取到最后的執(zhí)行結(jié)果。如果需要獲取執(zhí)行結(jié)果,就必須通過共享變量或線程通...

    zhaot 評論0 收藏0
  • FutureTask源碼解析(2)——深入理解FutureTask

    摘要:本文的源碼基于。人如其名,包含了和兩部分。而將一個(gè)任務(wù)的狀態(tài)設(shè)置成終止態(tài)只有三種方法我們將在下文的源碼解析中分析這三個(gè)方法。將棧中所有掛起的線程都喚醒后,下面就是執(zhí)行方法這個(gè)方法是一個(gè)空方 前言 系列文章目錄 有了上一篇對預(yù)備知識(shí)的了解之后,分析源碼就容易多了,本篇我們就直接來看看FutureTask的源碼。 本文的源碼基于JDK1.8。 Future和Task 在深入分析源碼之前,我...

    Harpsichord1207 評論0 收藏0
  • 系列文章目錄

    摘要:為了避免一篇文章的篇幅過長,于是一些比較大的主題就都分成幾篇來講了,這篇文章是筆者所有文章的目錄,將會(huì)持續(xù)更新,以給大家一個(gè)查看系列文章的入口。 前言 大家好,筆者是今年才開始寫博客的,寫作的初衷主要是想記錄和分享自己的學(xué)習(xí)經(jīng)歷。因?yàn)閷懽鞯臅r(shí)候發(fā)現(xiàn),為了弄懂一個(gè)知識(shí),不得不先去了解另外一些知識(shí),這樣以來,為了說明一個(gè)問題,就要把一系列知識(shí)都了解一遍,寫出來的文章就特別長。 為了避免一篇...

    lijy91 評論0 收藏0
  • 系列文章目錄

    摘要:為了避免一篇文章的篇幅過長,于是一些比較大的主題就都分成幾篇來講了,這篇文章是筆者所有文章的目錄,將會(huì)持續(xù)更新,以給大家一個(gè)查看系列文章的入口。 前言 大家好,筆者是今年才開始寫博客的,寫作的初衷主要是想記錄和分享自己的學(xué)習(xí)經(jīng)歷。因?yàn)閷懽鞯臅r(shí)候發(fā)現(xiàn),為了弄懂一個(gè)知識(shí),不得不先去了解另外一些知識(shí),這樣以來,為了說明一個(gè)問題,就要把一系列知識(shí)都了解一遍,寫出來的文章就特別長。 為了避免一篇...

    Yumenokanata 評論0 收藏0

發(fā)表評論

0條評論

最新活動(dòng)
閱讀需要支付1元查看
<