欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

ExecutorService Callable Future多線(xiàn)程返回結(jié)果原理解析

 更新時(shí)間:2022年09月26日 16:21:01   作者:朱季謙  
這篇文章主要為大家介紹了ExecutorService Callable Future多線(xiàn)程返回結(jié)果,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

正文

在并發(fā)多線(xiàn)程場(chǎng)景下,存在需要獲取各線(xiàn)程的異步執(zhí)行結(jié)果,這時(shí),就可以通過(guò)ExecutorService線(xiàn)程池結(jié)合Callable、Future來(lái)實(shí)現(xiàn)。

簡(jiǎn)單例子

public class ExecutorTest {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        ExecutorService executor = Executors.newSingleThreadExecutor();
        Callable callable = new MyCallable();
        Future future = executor.submit(callable);
        System.out.println("打印線(xiàn)程池返回值:" + future.get());
    }
}
class MyCallable implements Callable<String>{
    @Override
    public String call() throws Exception {
        return "測(cè)試返回值";
    }
}

執(zhí)行完成后,會(huì)打印出以下結(jié)果:

打印線(xiàn)程池返回值:測(cè)試返回值

可見(jiàn),線(xiàn)程池執(zhí)行完異步線(xiàn)程任務(wù),我們是可以獲取到異步線(xiàn)程里的返回值。

那么,ExecutorService、Callable、Future實(shí)現(xiàn)有返回結(jié)果的多線(xiàn)程是如何實(shí)現(xiàn)的呢?

首先,我們需要?jiǎng)?chuàng)建一個(gè)實(shí)現(xiàn)函數(shù)式接口Callable的類(lèi),該Callable接口只定義了一個(gè)被泛型修飾的call方法,這意味著,需要返回什么類(lèi)型的值可以由具體實(shí)現(xiàn)類(lèi)來(lái)定義——

@FunctionalInterface
public interface Callable<V> {
    V call() throws Exception;
}

因此,我自定義了一個(gè)實(shí)現(xiàn)Callable接口的類(lèi),該類(lèi)的重寫(xiě)了call方法,我們?cè)趫?zhí)行多線(xiàn)程時(shí)希望返回什么樣的結(jié)果,就可以在該重寫(xiě)的call方法定義。

class MyCallable implements Callable<String>{
    @Override
    public String call() throws Exception {
        return "測(cè)試返回值";
    }
}

在自定義的MyCallable類(lèi)中,我在call方法里設(shè)置一個(gè)很簡(jiǎn)單的String返回值 “測(cè)試返回值”,這意味著,我是希望在線(xiàn)程池執(zhí)行完異步線(xiàn)程任務(wù)時(shí),可以返回“測(cè)試返回值”這個(gè)字符串給我。

接下來(lái),我們就可以創(chuàng)建該MyCallable類(lèi)的對(duì)象,然后通過(guò)executor.submit(callable)丟到線(xiàn)程池里,線(xiàn)程池里會(huì)利用空閑線(xiàn)程來(lái)幫我們執(zhí)行一個(gè)異步線(xiàn)程任務(wù)。

ExecutorService executor = Executors.newSingleThreadExecutor();
Callable callable = new MyCallable();
Future future = executor.submit(callable);

值得注意一點(diǎn)是,若需要實(shí)現(xiàn)獲取線(xiàn)程返回值的效果,只能通過(guò)executor.submit(callable)去執(zhí)行,而不能通過(guò)executor.execute(Runnable command)執(zhí)行,因?yàn)閑xecutor.execute(Runnable command)只能傳入實(shí)現(xiàn)Runnable 接口的對(duì)象,但這類(lèi)對(duì)象是不具備返回線(xiàn)程效果的功能。

進(jìn)入到executor.submit(callable)底層,具體實(shí)現(xiàn)在AbstractExecutorService類(lèi)中。可以看到,執(zhí)行到submit方法內(nèi)部時(shí),會(huì)將我們傳進(jìn)來(lái)的new MyCallable()對(duì)象作為參數(shù)傳入到newTaskFor(task)方法里——

public <T> Future<T> submit(Callable<T> task) {
    if (task == null) throw new NullPointerException();
    RunnableFuture<T> ftask = newTaskFor(task);
    execute(ftask);
    return ftask;
}

這個(gè)newTaskFor(task)方法內(nèi)部具體實(shí)現(xiàn),是將new MyCallable()對(duì)象傳入構(gòu)造器中,生成了一個(gè)FutureTask對(duì)象。

protected <T> RunnableFuture<T> newTaskFor(Callable<T> callable) {
    return new FutureTask<T>(callable);
}

這個(gè)FutureTask對(duì)象實(shí)現(xiàn)RunableFuture接口,這個(gè)RunableFuture接口又繼承了Runnable,說(shuō)明FutureTask類(lèi)內(nèi)部會(huì)實(shí)現(xiàn)一個(gè)run方法,然后本身就可以當(dāng)做一個(gè)Runnable線(xiàn)程任務(wù),借助線(xiàn)程Thread(new FutureTask(.....)).start()方式開(kāi)啟一個(gè)新線(xiàn)程,去異步執(zhí)行其內(nèi)部實(shí)現(xiàn)的run方法邏輯。

異步執(zhí)行內(nèi)部實(shí)現(xiàn)run方法邏輯

public class FutureTask<V> implements RunnableFuture<V>{.....}
public interface RunnableFuture<V> extends Runnable, Future<V> {
    /**
     * Sets this Future to the result of its computation
     * unless it has been cancelled.
     */
    void run();
}

分析到這里,可以知道FutureTask的核心方法一定是run方法,線(xiàn)程執(zhí)行start方法后,最后會(huì)去調(diào)用FutureTask的run方法。在講解這個(gè)run方法前,我們先去看一下創(chuàng)建FutureTask的初始化構(gòu)造方法底層邏輯new FutureTask(callable)

public class FutureTask<V> implements RunnableFuture<V> {
private Callable<V> callable;
......//省略其余源碼
public FutureTask(Callable<V> callable) {
    if (callable == null)
        throw new NullPointerException();
    //通過(guò)構(gòu)造方法初始化Callable<V> callable賦值
    this.callable = callable;
    this.state = NEW;       // ensure visibility of callable
}
......//省略其余源碼
}

可以看到,F(xiàn)utureTask(Callable callable)構(gòu)造器,主要是將我們先前創(chuàng)建的new MyCallable()對(duì)象傳進(jìn)來(lái),賦值給FutureTask內(nèi)部定義的Callable callable引用,實(shí)現(xiàn)子類(lèi)對(duì)象指向父類(lèi)引用。這一點(diǎn)很關(guān)鍵,這就意味著,在初始化創(chuàng)建FutureTask對(duì)象后,我們是可以通過(guò)callable.call()來(lái)調(diào)用我們自定義設(shè)置可以返回“測(cè)試返回值”的call方法,這不就是我們希望在異步線(xiàn)程執(zhí)行完后能夠返回的值嗎?

我們不妨猜測(cè)一下整體返數(shù)主流程,在Thread(new FutureTask(.....)).start()開(kāi)啟一個(gè)線(xiàn)程后,當(dāng)線(xiàn)程獲得了CPU時(shí)間片,就會(huì)去執(zhí)行FutureTask對(duì)象里的run方法,這時(shí)run方法里可以通過(guò)callable.call()調(diào)用到我們自定義的MyCallable#call()方法,進(jìn)而得到方法返回值 “測(cè)試返回值”——到這一步,只需要將這個(gè)返回值賦值給FutureTask里某個(gè)定義的對(duì)象屬性,那么,在主線(xiàn)程在通過(guò)獲取FutureTask里被賦值的X對(duì)象屬性值,不就可以拿到返回字符串值 “測(cè)試返回值”了嗎?

實(shí)現(xiàn)上,主體流程確實(shí)是這樣,只不過(guò)忽略了一些細(xì)節(jié)而已。

FutureTask的run方法

public void run() {
    //如果狀態(tài)不是NEW或者設(shè)置runner為當(dāng)前線(xiàn)程時(shí),說(shuō)明FutureTask任務(wù)已經(jīng)取消,無(wú)法繼續(xù)執(zhí)行
    if (state != NEW ||
        !UNSAFE.compareAndSwapObject(this, runnerOffset,
                                     null, Thread.currentThread()))
        return;
    try {
        //在該文中,callable被賦值為指向我們定義的new MyCallable()對(duì)象引用
        Callable<V> c = callable;
        if (c != null && state == NEW) {
            V result;
            boolean ran;
            try {
                //c.call最后會(huì)調(diào)用new MyCallable()的call()方法,得到字符串返回值“測(cè)試返回值”給result
                result = c.call();
                ran = true;
            } catch (Throwable ex) {
                result = null;
                ran = false;
                setException(ex);
            }
            //正常執(zhí)行完c.call()方法時(shí),ran值為true,說(shuō)明會(huì)執(zhí)行set(result)方法。
            if (ran)
                set(result);
        }
    } finally {
        // runner must be non-null until state is settled to
        // prevent concurrent calls to run()
        runner = null;
        // state must be re-read after nulling runner to prevent
        // leaked interrupts
        int s = state;
        if (s >= INTERRUPTING)
            handlePossibleCancellationInterrupt(s);
    }
}

根據(jù)以上源碼簡(jiǎn)單分析,可以看到run方法當(dāng)中,最終確實(shí)會(huì)執(zhí)行new MyCallable()的call()方法,得到字符串返回值“測(cè)試返回值”給result,然后執(zhí)行set(result)方法,根據(jù)set方法名就不難猜出,這是一個(gè)會(huì)賦值給某個(gè)字段的方法。

這里分析會(huì)忽略一些狀態(tài)值的講解,這塊會(huì)包括線(xiàn)程的取消、終止等內(nèi)容,后面我會(huì)出一片專(zhuān)門(mén)針對(duì)FutureTask源碼分析的文章再介紹,本文主要還是介紹異步線(xiàn)程返回結(jié)果的主要原理。

set(result)方法

沿著以上分析,追蹤至set(result)方法里——

protected void set(V v) {
    //通過(guò)CAS原子操作,將運(yùn)行的線(xiàn)程設(shè)置為COMPLETING,說(shuō)明線(xiàn)程已經(jīng)執(zhí)行完成中
    if (UNSAFE.compareAndSwapInt(this, stateOffset, NEW, COMPLETING)) {
        //若CAS原子比較賦值成功,說(shuō)明線(xiàn)程可以被正常執(zhí)行完成的話(huà),然后將result結(jié)果值賦值給outcome
        outcome = v;
        //線(xiàn)程正常完成結(jié)束
        UNSAFE.putOrderedInt(this, stateOffset, NORMAL); // final state
        finishCompletion();
    }
}

這個(gè)方法的主要是,若該線(xiàn)程執(zhí)行能夠正常完成話(huà),就將得到的返回值賦值給outcome,這個(gè)outcome是FutureTask的一個(gè)Object變量——

private Object outcome;

至此,就完成了流程的這一步——

最后,就是執(zhí)行主線(xiàn)程的根據(jù)ftask.get()獲取執(zhí)行完成的值,這個(gè)get可以設(shè)置超時(shí)時(shí)間,例如 ftask.get(2,TimeUnit.SECONDS)表示超過(guò)2秒還沒(méi)有獲取到線(xiàn)程返回值的話(huà),就直接結(jié)束該get方法,繼續(xù)主線(xiàn)程往下執(zhí)行。

System.out.println("打印線(xiàn)程池返回值:" + ftask.get(2,TimeUnit.SECONDS));

進(jìn)入到get方法,可以看到當(dāng)狀態(tài)在s <= COMPLETING時(shí),表示任務(wù)還沒(méi)有執(zhí)行完,就會(huì)去執(zhí)行awaitDone(false, 0L)方法,這個(gè)方法表示,將一直做死循環(huán)等待線(xiàn)程執(zhí)行完成,才會(huì)跳出等待循環(huán)繼續(xù)往下走。若設(shè)置了超時(shí)時(shí)間,例如ftask.get(2,TimeUnit.SECONDS)),就會(huì)在awaitDone方法循環(huán)至2秒,在2秒內(nèi)發(fā)現(xiàn)線(xiàn)程狀態(tài)被設(shè)置為正常完成時(shí),就會(huì)跳出循環(huán),若2秒后線(xiàn)程沒(méi)有執(zhí)行完成,也會(huì)強(qiáng)制跳出循環(huán)了,但這種情況將無(wú)法獲取到線(xiàn)程結(jié)果值。

public V get() throws InterruptedException, ExecutionException {
    int s = state;
    if (s &lt;= COMPLETING)
        //循環(huán)等待線(xiàn)程執(zhí)行狀態(tài)
        s = awaitDone(false, 0L);
    return report(s);
}

最后就是report(s)方法,可以看到outcome值最終賦值給Object x,若s==NORMAL表示線(xiàn)程任務(wù)已經(jīng)正常完成結(jié)束,就可以根據(jù)我們定義的類(lèi)型進(jìn)行泛型轉(zhuǎn)換返回,我們定義的是String字符串類(lèi)型,故而會(huì)返回字符串值,也就是 “測(cè)試返回值”。

private V report(int s) throws ExecutionException {
    Object x = outcome;
    if (s == NORMAL)
        //返回線(xiàn)程任務(wù)執(zhí)行結(jié)果
        return (V)x;
    if (s >= CANCELLED)
        throw new CancellationException();
    throw new ExecutionException((Throwable)x);
}

你看,最后就能獲取到了異步線(xiàn)程執(zhí)行的結(jié)果返回給main主線(xiàn)程——

以上就是執(zhí)行線(xiàn)程任務(wù)run方法后,如何將線(xiàn)程任務(wù)結(jié)果返回給主線(xiàn)程,其實(shí),還少一個(gè)地方補(bǔ)充,就是如何將FutureTask任務(wù)丟給線(xiàn)程執(zhí)行,我們這里用到了線(xiàn)程池, 但是execute(ftask)底層同樣是使用一個(gè)了線(xiàn)程通過(guò)執(zhí)行start方法開(kāi)啟一個(gè)線(xiàn)程,這個(gè)新運(yùn)行的線(xiàn)程最終會(huì)執(zhí)行FutureTask的run方法。

public <T> Future<T> submit(Callable<T> task) {
    if (task == null) throw new NullPointerException();
    RunnableFuture<T> ftask = newTaskFor(task);
    execute(ftask);
    return ftask;
}

可以簡(jiǎn)單優(yōu)化下,直接用一個(gè)線(xiàn)程演示該案例,這樣看著更好理解些,當(dāng)時(shí),生產(chǎn)上是不會(huì)有這樣直接用一個(gè)線(xiàn)程來(lái)執(zhí)行的,更多是通過(guò)原生線(xiàn)程池——

public static void main(String[] args) throws Exception{
    Callable callable = new MyCallable();
    RunnableFuture<String> ftask = new FutureTask<String>(callable);
    new Thread(ftask).start();
    System.out.println("打印線(xiàn)程池返回值:" + ftask.get());
}

以上就是ExecutorService Callable Future多線(xiàn)程返回結(jié)果原理解析的詳細(xì)內(nèi)容,更多關(guān)于ExecutorService Callable Future多線(xiàn)程的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Java全面細(xì)致講解Wrapper的使用

    Java全面細(xì)致講解Wrapper的使用

    在封裝中有一種特殊的類(lèi),能夠把基本的數(shù)據(jù)類(lèi)型進(jìn)行轉(zhuǎn)換來(lái)方便實(shí)際的使用。我們?cè)谥疤岬降囊恍?shù)據(jù)類(lèi)型,最明顯的特征是所有字母為小寫(xiě)狀態(tài),那么經(jīng)過(guò)Wrapper的包裝后,首字母就變成了大寫(xiě)。下面我們就這種特殊的封裝類(lèi)Wrapper的使用
    2022-05-05
  • SpringBoot環(huán)境下junit單元測(cè)試速度優(yōu)化方式

    SpringBoot環(huán)境下junit單元測(cè)試速度優(yōu)化方式

    這篇文章主要介紹了SpringBoot環(huán)境下junit單元測(cè)試速度優(yōu)化方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • Springboot使用jxls實(shí)現(xiàn)同sheet多個(gè)列表展示

    Springboot使用jxls實(shí)現(xiàn)同sheet多個(gè)列表展示

    這篇文章主要介紹了Springboot使用jxls實(shí)現(xiàn)同sheet多個(gè)列表展示,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-08-08
  • mybatis的映射xml中動(dòng)態(tài)設(shè)置orderby方式

    mybatis的映射xml中動(dòng)態(tài)設(shè)置orderby方式

    這篇文章主要介紹了mybatis的映射xml中動(dòng)態(tài)設(shè)置orderby方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-11-11
  • SpringMVC表單標(biāo)簽使用詳解

    SpringMVC表單標(biāo)簽使用詳解

    這篇文章主要為大家詳細(xì)介紹了SpringMVC表單標(biāo)簽的使用方法,教大家如何用Spring封裝的一系列表單標(biāo)簽
    2017-03-03
  • ssm框架上傳圖片保存到本地和數(shù)據(jù)庫(kù)示例

    ssm框架上傳圖片保存到本地和數(shù)據(jù)庫(kù)示例

    本篇文章主要介紹了ssm框架上傳圖片保存到本地和數(shù)據(jù)庫(kù)示例,主要使用了Spring+SpringMVC+MyBatis框架集合,有興趣的可以了解一下。
    2017-03-03
  • 詳解JAVA設(shè)計(jì)模式之適配器模式

    詳解JAVA設(shè)計(jì)模式之適配器模式

    這篇文章主要介紹了JAVA設(shè)計(jì)模式之適配器模式的的相關(guān)資料,文中示例代碼非常詳細(xì),供大家參考和學(xué)習(xí),感興趣的朋友可以了解
    2020-06-06
  • 最新評(píng)論