日日操夜夜添-日日操影院-日日草夜夜操-日日干干-精品一区二区三区波多野结衣-精品一区二区三区高清免费不卡

公告:魔扣目錄網(wǎng)為廣大站長(zhǎng)提供免費(fèi)收錄網(wǎng)站服務(wù),提交前請(qǐng)做好本站友鏈:【 網(wǎng)站目錄:http://www.ylptlb.cn 】, 免友鏈快審服務(wù)(50元/站),

點(diǎn)擊這里在線咨詢客服
新站提交
  • 網(wǎng)站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會(huì)員:747

 

一個(gè)示例回顧Future

一些業(yè)務(wù)場(chǎng)景我們需要使用多線程異步執(zhí)行任務(wù),加快任務(wù)執(zhí)行速度。

JDK5新增了Future接口,用于描述一個(gè)異步計(jì)算的結(jié)果。

雖然 Future 以及相關(guān)使用方法提供了異步執(zhí)行任務(wù)的能力,但是對(duì)于結(jié)果的獲取卻是很不方便,我們必須使用Future.get()的方式阻塞調(diào)用線程,或者使用輪詢方式判斷 Future.isDone 任務(wù)是否結(jié)束,再獲取結(jié)果。

這兩種處理方式都不是很優(yōu)雅,相關(guān)代碼如下:

    @Test
    public void testFuture() throws ExecutionException, InterruptedException {
        ExecutorService executorService = Executors.newFixedThreadPool(5);
        Future<String> future = executorService.submit(() -> {
            Thread.sleep(2000);
            return "hello";
        });
        System.out.println(future.get());
        System.out.println("end");
    }

與此同時(shí),F(xiàn)uture無(wú)法解決多個(gè)異步任務(wù)需要相互依賴的場(chǎng)景,簡(jiǎn)單點(diǎn)說(shuō)就是,主線程需要等待子線程任務(wù)執(zhí)行完畢之后在進(jìn)行執(zhí)行,這個(gè)時(shí)候你可能想到了「CountDownLatch」,沒(méi)錯(cuò)確實(shí)可以解決,代碼如下。

這里定義兩個(gè)Future,第一個(gè)通過(guò)用戶id獲取用戶信息,第二個(gè)通過(guò)商品id獲取商品信息。

  @Test
    public void testCountDownLatch() throws InterruptedException, ExecutionException {
        ExecutorService executorService = Executors.newFixedThreadPool(5);
        CountDownLatch downLatch = new CountDownLatch(2);
        long startTime = System.currentTimeMillis();
        Future<String> userFuture = executorService.submit(() -> {
            //模擬查詢商品耗時(shí)500毫秒
            Thread.sleep(500);
            downLatch.countDown();
            return "用戶A";
        });

        Future<String> goodsFuture = executorService.submit(() -> {
            //模擬查詢商品耗時(shí)500毫秒
            Thread.sleep(400);
            downLatch.countDown();
            return "商品A";
        });

        downLatch.await();
        //模擬主程序耗時(shí)時(shí)間
        Thread.sleep(600);
        System.out.println("獲取用戶信息:" + userFuture.get());
        System.out.println("獲取商品信息:" + goodsFuture.get());
        System.out.println("總共用時(shí)" + (System.currentTimeMillis() - startTime) + "ms");

    }

「運(yùn)行結(jié)果」

獲取用戶信息:用戶A
獲取商品信息:商品A
總共用時(shí)1110ms

從運(yùn)行結(jié)果可以看出結(jié)果都已經(jīng)獲取,而且如果我們不用異步操作,執(zhí)行時(shí)間應(yīng)該是:500+400+600 = 1500,用異步操作后實(shí)際只用1110。

但是JAVA8以后我不在認(rèn)為這是一種優(yōu)雅的解決方式,接下來(lái)來(lái)了解下CompletableFuture的使用。

通過(guò)CompletableFuture實(shí)現(xiàn)上面示例

 @Test
    public void testCompletableInfo() throws InterruptedException, ExecutionException {
        long startTime = System.currentTimeMillis();
  
          //調(diào)用用戶服務(wù)獲取用戶基本信息
          CompletableFuture<String> userFuture = CompletableFuture.supplyAsync(() ->
                  //模擬查詢商品耗時(shí)500毫秒
          {
              try {
                  Thread.sleep(500);
              } catch (InterruptedException e) {
                  e.printStackTrace();
              }
              return "用戶A";
          });
  
          //調(diào)用商品服務(wù)獲取商品基本信息
          CompletableFuture<String> goodsFuture = CompletableFuture.supplyAsync(() ->
                  //模擬查詢商品耗時(shí)500毫秒
          {
              try {
                  Thread.sleep(400);
              } catch (InterruptedException e) {
                  e.printStackTrace();
              }
              return "商品A";
          });
  
          System.out.println("獲取用戶信息:" + userFuture.get());
          System.out.println("獲取商品信息:" + goodsFuture.get());
  
          //模擬主程序耗時(shí)時(shí)間
          Thread.sleep(600);
          System.out.println("總共用時(shí)" + (System.currentTimeMillis() - startTime) + "ms");
    }

運(yùn)行結(jié)果

獲取用戶信息:用戶A
獲取商品信息:商品A
總共用時(shí)1112ms

通過(guò)CompletableFuture可以很輕松的實(shí)現(xiàn)CountDownLatch的功能,你以為這就結(jié)束了,遠(yuǎn)遠(yuǎn)不止,CompletableFuture比這要強(qiáng)多了。

比如可以實(shí)現(xiàn):任務(wù)1執(zhí)行完了再執(zhí)行任務(wù)2,甚至任務(wù)1執(zhí)行的結(jié)果,作為任務(wù)2的入?yún)?shù)等等強(qiáng)大功能,下面就來(lái)學(xué)學(xué)CompletableFuture的API。

CompletableFuture創(chuàng)建方式

1、常用的4種創(chuàng)建方式

CompletableFuture源碼中有四個(gè)靜態(tài)方法用來(lái)執(zhí)行異步任務(wù)

public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier){..}
public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier,Executor executor){..}
public static CompletableFuture<Void> runAsync(Runnable runnable){..}
public static CompletableFuture<Void> runAsync(Runnable runnable,Executor executor){..}

一般我們用上面的靜態(tài)方法來(lái)創(chuàng)建CompletableFuture,這里也解釋下他們的區(qū)別:

  • 「supplyAsync」執(zhí)行任務(wù),支持返回值。
  • 「runAsync」執(zhí)行任務(wù),沒(méi)有返回值。

「supplyAsync方法」

//使用默認(rèn)內(nèi)置線程池ForkJoinPool.commonPool(),根據(jù)supplier構(gòu)建執(zhí)行任務(wù)
public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier)
//自定義線程,根據(jù)supplier構(gòu)建執(zhí)行任務(wù)
public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier, Executor executor)

「runAsync方法」

//使用默認(rèn)內(nèi)置線程池ForkJoinPool.commonPool(),根據(jù)runnable構(gòu)建執(zhí)行任務(wù)
public static CompletableFuture<Void> runAsync(Runnable runnable) 
//自定義線程,根據(jù)runnable構(gòu)建執(zhí)行任務(wù)
public static CompletableFuture<Void> runAsync(Runnable runnable,  Executor executor)

2、結(jié)果獲取的4種方式

對(duì)于結(jié)果的獲取CompltableFuture類提供了四種方式

//方式一
public T get()
//方式二
public T get(long timeout, TimeUnit unit)
//方式三
public T getNow(T valueIfAbsent)
//方式四
public T join()

說(shuō)明

  • 「get()和get(long timeout, TimeUnit unit)」 => 在Future中就已經(jīng)提供了,后者提供超時(shí)處理,如果在指定時(shí)間內(nèi)未獲取結(jié)果將拋出超時(shí)異常
  • 「getNow」 => 立即獲取結(jié)果不阻塞,結(jié)果計(jì)算已完成將返回結(jié)果或計(jì)算過(guò)程中的異常,如果未計(jì)算完成將返回設(shè)定的valueIfAbsent值
  • 「join」 => 方法里不會(huì)拋出異常

示例

  @Test
    public void testCompletableGet() throws InterruptedException, ExecutionException {

        CompletableFuture<String> cp1 = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return "商品A";
        });

        // getNow方法測(cè)試 
        System.out.println(cp1.getNow("商品B"));

        //join方法測(cè)試 
        CompletableFuture<Integer> cp2 = CompletableFuture.supplyAsync((() -> 1 / 0));
        System.out.println(cp2.join());
       System.out.println("-----------------------------------------------------");
        //get方法測(cè)試
        CompletableFuture<Integer> cp3 = CompletableFuture.supplyAsync((() -> 1 / 0));
        System.out.println(cp3.get());
    }

「運(yùn)行結(jié)果」

  • 第一個(gè)執(zhí)行結(jié)果為 「商品B」,因?yàn)橐人?秒結(jié)果不能立即獲取
  • join方法獲取結(jié)果方法里不會(huì)拋異常,但是執(zhí)行結(jié)果會(huì)拋異常,拋出的異常為CompletionException
  • get方法獲取結(jié)果方法里將拋出異常,執(zhí)行結(jié)果拋出的異常為ExecutionException

異步回調(diào)方法

 

1、thenRun/thenRunAsync

通俗點(diǎn)講就是,「做完第一個(gè)任務(wù)后,再做第二個(gè)任務(wù),第二個(gè)任務(wù)也沒(méi)有返回值」

示例

 @Test
    public void testCompletableThenRunAsync() throws InterruptedException, ExecutionException {
        long startTime = System.currentTimeMillis();
        
        CompletableFuture<Void> cp1 = CompletableFuture.runAsync(() -> {
            try {
                //執(zhí)行任務(wù)A
                Thread.sleep(600);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

        });

        CompletableFuture<Void> cp2 =  cp1.thenRun(() -> {
            try {
                //執(zhí)行任務(wù)B
                Thread.sleep(400);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });

        // get方法測(cè)試
        System.out.println(cp2.get());

        //模擬主程序耗時(shí)時(shí)間
        Thread.sleep(600);
        System.out.println("總共用時(shí)" + (System.currentTimeMillis() - startTime) + "ms");
    }
    
    //運(yùn)行結(jié)果  
    /**
     *  null
     *  總共用時(shí)1610ms
     */

「thenRun 和thenRunAsync有什么區(qū)別呢?」

如果你執(zhí)行第一個(gè)任務(wù)的時(shí)候,傳入了一個(gè)自定義線程池:

  • 調(diào)用thenRun方法執(zhí)行第二個(gè)任務(wù)時(shí),則第二個(gè)任務(wù)和第一個(gè)任務(wù)是共用同一個(gè)線程池。
  • 調(diào)用thenRunAsync執(zhí)行第二個(gè)任務(wù)時(shí),則第一個(gè)任務(wù)使用的是你自己傳入的線程池,第二個(gè)任務(wù)使用的是ForkJoin線程池。

說(shuō)明: 后面介紹的thenAccept和thenAcceptAsync,thenApply和thenApplyAsync等,它們之間的區(qū)別也是這個(gè)。

2、thenAccept/thenAcceptAsync

第一個(gè)任務(wù)執(zhí)行完成后,執(zhí)行第二個(gè)回調(diào)方法任務(wù),會(huì)將該任務(wù)的執(zhí)行結(jié)果,作為入?yún)?/strong>,傳遞到回調(diào)方法中,但是回調(diào)方法是沒(méi)有返回值的。

示例

    @Test
    public void testCompletableThenAccept() throws ExecutionException, InterruptedException {
        long startTime = System.currentTimeMillis();
        CompletableFuture<String> cp1 = CompletableFuture.supplyAsync(() -> {
            return "dev";

        });
        CompletableFuture<Void> cp2 =  cp1.thenAccept((a) -> {
            System.out.println("上一個(gè)任務(wù)的返回結(jié)果為: " + a);
        });
     
        cp2.get();
    }

3、 thenApply/thenApplyAsync

表示第一個(gè)任務(wù)執(zhí)行完成后,執(zhí)行第二個(gè)回調(diào)方法任務(wù),會(huì)將該任務(wù)的執(zhí)行結(jié)果,作為入?yún)ⅲ瑐鬟f到回調(diào)方法中,并且回調(diào)方法是有返回值的。

示例

   @Test
    public void testCompletableThenApply() throws ExecutionException, InterruptedException {
        CompletableFuture<String> cp1 = CompletableFuture.supplyAsync(() -> {
            return "dev";

        }).thenApply((a) -> {
            if(Objects.equals(a,"dev")){
                return "dev";
            }
            return "prod";
        });

        System.out.println("當(dāng)前環(huán)境為:" + cp1.get());

        //輸出: 當(dāng)前環(huán)境為:dev
    }

異常回調(diào)

當(dāng)CompletableFuture的任務(wù)不論是正常完成還是出現(xiàn)異常它都會(huì)調(diào)用「whenComplete」這回調(diào)函數(shù)。

  • 「正常完成」:whenComplete返回結(jié)果和上級(jí)任務(wù)一致,異常為null;
  • 「出現(xiàn)異常」:whenComplete返回結(jié)果為null,異常為上級(jí)任務(wù)的異常;

即調(diào)用get()時(shí),正常完成時(shí)就獲取到結(jié)果,出現(xiàn)異常時(shí)就會(huì)拋出異常,需要你處理該異常。

下面來(lái)看看示例

1、只用whenComplete

    @Test
    public void testCompletableWhenComplete() throws ExecutionException, InterruptedException {
        CompletableFuture<Double> future = CompletableFuture.supplyAsync(() -> {

            if (Math.random() < 0.5) {
                throw new RuntimeException("出錯(cuò)了");
            }
            System.out.println("正常結(jié)束");
            return 0.11;

        }).whenComplete((aDouble, throwable) -> {
            if (aDouble == null) {
                System.out.println("whenComplete aDouble is null");
            } else {
                System.out.println("whenComplete aDouble is " + aDouble);
            }
            if (throwable == null) {
                System.out.println("whenComplete throwable is null");
            } else {
                System.out.println("whenComplete throwable is " + throwable.getMessage());
            }
        });
        System.out.println("最終返回的結(jié)果 = " + future.get());
    }

正常完成,沒(méi)有異常時(shí):

正常結(jié)束
whenComplete aDouble is 0.11
whenComplete throwable is null
最終返回的結(jié)果 = 0.11

出現(xiàn)異常時(shí):get()會(huì)拋出異常

whenComplete aDouble is null
whenComplete throwable is java.lang.RuntimeException: 出錯(cuò)了

java.util.concurrent.ExecutionException: java.lang.RuntimeException: 出錯(cuò)了
 at java.util.concurrent.CompletableFuture.reportGet(CompletableFuture.java:357)
 at java.util.concurrent.CompletableFuture.get(CompletableFuture.java:1895)

2、whenComplete + exceptionally示例

 @Test
    public void testWhenCompleteExceptionally() throws ExecutionException, InterruptedException {
        CompletableFuture<Double> future = CompletableFuture.supplyAsync(() -> {
            if (Math.random() < 0.5) {
                throw new RuntimeException("出錯(cuò)了");
            }
            System.out.println("正常結(jié)束");
            return 0.11;

        }).whenComplete((aDouble, throwable) -> {
            if (aDouble == null) {
                System.out.println("whenComplete aDouble is null");
            } else {
                System.out.println("whenComplete aDouble is " + aDouble);
            }
            if (throwable == null) {
                System.out.println("whenComplete throwable is null");
            } else {
                System.out.println("whenComplete throwable is " + throwable.getMessage());
            }
        }).exceptionally((throwable) -> {
            System.out.println("exceptionally中異常:" + throwable.getMessage());
            return 0.0;
        });

        System.out.println("最終返回的結(jié)果 = " + future.get());
    }

當(dāng)出現(xiàn)異常時(shí),exceptionally中會(huì)捕獲該異常,給出默認(rèn)返回值0.0。

whenComplete aDouble is null
whenComplete throwable is java.lang.RuntimeException: 出錯(cuò)了
exceptionally中異常:java.lang.RuntimeException: 出錯(cuò)了
最終返回的結(jié)果 = 0.0


 

多任務(wù)組合回調(diào)

 

1、AND組合關(guān)系

thenCombine / thenAcceptBoth / runAfterBoth都表示:「當(dāng)任務(wù)一和任務(wù)二都完成再執(zhí)行任務(wù)三」

區(qū)別在于:

  • 「runAfterBoth」 不會(huì)把執(zhí)行結(jié)果當(dāng)做方法入?yún)ⅲ覜](méi)有返回值
  • 「thenAcceptBoth」: 會(huì)將兩個(gè)任務(wù)的執(zhí)行結(jié)果作為方法入?yún)ⅲ瑐鬟f到指定方法中,且無(wú)返回值
  • 「thenCombine」:會(huì)將兩個(gè)任務(wù)的執(zhí)行結(jié)果作為方法入?yún)ⅲ瑐鬟f到指定方法中,且有返回值

示例

    @Test
    public void testCompletableThenCombine() throws ExecutionException, InterruptedException {
        //創(chuàng)建線程池
        ExecutorService executorService = Executors.newFixedThreadPool(10);
        //開(kāi)啟異步任務(wù)1
        CompletableFuture<Integer> task = CompletableFuture.supplyAsync(() -> {
            System.out.println("異步任務(wù)1,當(dāng)前線程是:" + Thread.currentThread().getId());
            int result = 1 + 1;
            System.out.println("異步任務(wù)1結(jié)束");
            return result;
        }, executorService);

        //開(kāi)啟異步任務(wù)2
        CompletableFuture<Integer> task2 = CompletableFuture.supplyAsync(() -> {
            System.out.println("異步任務(wù)2,當(dāng)前線程是:" + Thread.currentThread().getId());
            int result = 1 + 1;
            System.out.println("異步任務(wù)2結(jié)束");
            return result;
        }, executorService);

        //任務(wù)組合
        CompletableFuture<Integer> task3 = task.thenCombineAsync(task2, (f1, f2) -> {
            System.out.println("執(zhí)行任務(wù)3,當(dāng)前線程是:" + Thread.currentThread().getId());
            System.out.println("任務(wù)1返回值:" + f1);
            System.out.println("任務(wù)2返回值:" + f2);
            return f1 + f2;
        }, executorService);

        Integer res = task3.get();
        System.out.println("最終結(jié)果:" + res);
    }

「運(yùn)行結(jié)果」

異步任務(wù)1,當(dāng)前線程是:17
異步任務(wù)1結(jié)束
異步任務(wù)2,當(dāng)前線程是:18
異步任務(wù)2結(jié)束
執(zhí)行任務(wù)3,當(dāng)前線程是:19
任務(wù)1返回值:2
任務(wù)2返回值:2
最終結(jié)果:4

2、OR組合關(guān)系

applyToEither / acceptEither / runAfterEither 都表示:「兩個(gè)任務(wù),只要有一個(gè)任務(wù)完成,就執(zhí)行任務(wù)三」

區(qū)別在于:

  • 「runAfterEither」:不會(huì)把執(zhí)行結(jié)果當(dāng)做方法入?yún)ⅲ覜](méi)有返回值
  • 「acceptEither」: 會(huì)將已經(jīng)執(zhí)行完成的任務(wù),作為方法入?yún)ⅲ瑐鬟f到指定方法中,且無(wú)返回值
  • 「applyToEither」:會(huì)將已經(jīng)執(zhí)行完成的任務(wù),作為方法入?yún)ⅲ瑐鬟f到指定方法中,且有返回值

示例

  @Test
    public void testCompletableEitherAsync() {
        //創(chuàng)建線程池
        ExecutorService executorService = Executors.newFixedThreadPool(10);
        //開(kāi)啟異步任務(wù)1
        CompletableFuture<Integer> task = CompletableFuture.supplyAsync(() -> {
            System.out.println("異步任務(wù)1,當(dāng)前線程是:" + Thread.currentThread().getId());

            int result = 1 + 1;
            System.out.println("異步任務(wù)1結(jié)束");
            return result;
        }, executorService);

        //開(kāi)啟異步任務(wù)2
        CompletableFuture<Integer> task2 = CompletableFuture.supplyAsync(() -> {
            System.out.println("異步任務(wù)2,當(dāng)前線程是:" + Thread.currentThread().getId());
            int result = 1 + 2;
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("異步任務(wù)2結(jié)束");
            return result;
        }, executorService);

        //任務(wù)組合
        task.acceptEitherAsync(task2, (res) -> {
            System.out.println("執(zhí)行任務(wù)3,當(dāng)前線程是:" + Thread.currentThread().getId());
            System.out.println("上一個(gè)任務(wù)的結(jié)果為:"+res);
        }, executorService);
    }

運(yùn)行結(jié)果

//通過(guò)結(jié)果可以看出,異步任務(wù)2都沒(méi)有執(zhí)行結(jié)束,任務(wù)3獲取的也是1的執(zhí)行結(jié)果
異步任務(wù)1,當(dāng)前線程是:17
異步任務(wù)1結(jié)束
異步任務(wù)2,當(dāng)前線程是:18
執(zhí)行任務(wù)3,當(dāng)前線程是:19
上一個(gè)任務(wù)的結(jié)果為:2

注意

如果把上面的核心線程數(shù)改為1也就是

 ExecutorService executorService = Executors.newFixedThreadPool(1);

運(yùn)行結(jié)果就是下面的了,會(huì)發(fā)現(xiàn)根本沒(méi)有執(zhí)行任務(wù)3,顯然是任務(wù)3直接被丟棄了。

異步任務(wù)1,當(dāng)前線程是:17
異步任務(wù)1結(jié)束
異步任務(wù)2,當(dāng)前線程是:17

3、多任務(wù)組合

  • 「allOf」:等待所有任務(wù)完成
  • 「anyOf」:只要有一個(gè)任務(wù)完成

示例

allOf:等待所有任務(wù)完成

 @Test
    public void testCompletableAallOf() throws ExecutionException, InterruptedException {
        //創(chuàng)建線程池
        ExecutorService executorService = Executors.newFixedThreadPool(10);
        //開(kāi)啟異步任務(wù)1
        CompletableFuture<Integer> task = CompletableFuture.supplyAsync(() -> {
            System.out.println("異步任務(wù)1,當(dāng)前線程是:" + Thread.currentThread().getId());
            int result = 1 + 1;
            System.out.println("異步任務(wù)1結(jié)束");
            return result;
        }, executorService);

        //開(kāi)啟異步任務(wù)2
        CompletableFuture<Integer> task2 = CompletableFuture.supplyAsync(() -> {
            System.out.println("異步任務(wù)2,當(dāng)前線程是:" + Thread.currentThread().getId());
            int result = 1 + 2;
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("異步任務(wù)2結(jié)束");
            return result;
        }, executorService);

        //開(kāi)啟異步任務(wù)3
        CompletableFuture<Integer> task3 = CompletableFuture.supplyAsync(() -> {
            System.out.println("異步任務(wù)3,當(dāng)前線程是:" + Thread.currentThread().getId());
            int result = 1 + 3;
            try {
                Thread.sleep(4000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("異步任務(wù)3結(jié)束");
            return result;
        }, executorService);

        //任務(wù)組合
        CompletableFuture<Void> allOf = CompletableFuture.allOf(task, task2, task3);

        //等待所有任務(wù)完成
        allOf.get();
        //獲取任務(wù)的返回結(jié)果
        System.out.println("task結(jié)果為:" + task.get());
        System.out.println("task2結(jié)果為:" + task2.get());
        System.out.println("task3結(jié)果為:" + task3.get());
    }

anyOf: 只要有一個(gè)任務(wù)完成

    @Test
    public void testCompletableAnyOf() throws ExecutionException, InterruptedException {
        //創(chuàng)建線程池
        ExecutorService executorService = Executors.newFixedThreadPool(10);
        //開(kāi)啟異步任務(wù)1
        CompletableFuture<Integer> task = CompletableFuture.supplyAsync(() -> {
            int result = 1 + 1;
            return result;
        }, executorService);

        //開(kāi)啟異步任務(wù)2
        CompletableFuture<Integer> task2 = CompletableFuture.supplyAsync(() -> {
            int result = 1 + 2;
            return result;
        }, executorService);

        //開(kāi)啟異步任務(wù)3
        CompletableFuture<Integer> task3 = CompletableFuture.supplyAsync(() -> {
            int result = 1 + 3;
            return result;
        }, executorService);

        //任務(wù)組合
        CompletableFuture<Object> anyOf = CompletableFuture.anyOf(task, task2, task3);
        //只要有一個(gè)有任務(wù)完成
        Object o = anyOf.get();
        System.out.println("完成的任務(wù)的結(jié)果:" + o);
    }

CompletableFuture使用有哪些注意點(diǎn)

 

CompletableFuture 使我們的異步編程更加便利的、代碼更加優(yōu)雅的同時(shí),我們也要關(guān)注下它,使用的一些注意點(diǎn)。

1、Future需要獲取返回值,才能獲取異常信息

  @Test
    public void testWhenCompleteExceptionally() {
        CompletableFuture<Double> future = CompletableFuture.supplyAsync(() -> {
            if (1 == 1) {
                throw new RuntimeException("出錯(cuò)了");
            }
            return 0.11;
        });

        //如果不加 get()方法這一行,看不到異常信息
        //future.get();
    }

Future需要獲取返回值,才能獲取到異常信息。如果不加 get()/join()方法,看不到異常信息。

小伙伴們使用的時(shí)候,注意一下哈,考慮是否加try...catch...或者使用exceptionally方法。

2、CompletableFuture的get()方法是阻塞的

CompletableFuture的get()方法是阻塞的,如果使用它來(lái)獲取異步調(diào)用的返回值,需要添加超時(shí)時(shí)間。

//反例
 CompletableFuture.get();
//正例
CompletableFuture.get(5, TimeUnit.SECONDS);

3、不建議使用默認(rèn)線程池

CompletableFuture代碼中又使用了默認(rèn)的「ForkJoin線程池」,處理的線程個(gè)數(shù)是電腦「CPU核數(shù)-1」。在大量請(qǐng)求過(guò)來(lái)的時(shí)候,處理邏輯復(fù)雜的話,響應(yīng)會(huì)很慢。一般建議使用自定義線程池,優(yōu)化線程池配置參數(shù)。

4、自定義線程池時(shí),注意飽和策略

CompletableFuture的get()方法是阻塞的,我們一般建議使用future.get(5, TimeUnit.SECONDS)。并且一般建議使用自定義線程池。

但是如果線程池拒絕策略是DiscardPolicy或者DiscardOldestPolicy,當(dāng)線程池飽和時(shí),會(huì)直接丟棄任務(wù),不會(huì)拋棄異常。因此建議,CompletableFuture線程池策略最好使用AbortPolicy,然后耗時(shí)的異步線程,做好線程池隔離哈。

分享到:
標(biāo)簽:CompletableFuture
用戶無(wú)頭像

網(wǎng)友整理

注冊(cè)時(shí)間:

網(wǎng)站:5 個(gè)   小程序:0 個(gè)  文章:12 篇

  • 51998

    網(wǎng)站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會(huì)員

趕快注冊(cè)賬號(hào),推廣您的網(wǎng)站吧!
最新入駐小程序

數(shù)獨(dú)大挑戰(zhàn)2018-06-03

數(shù)獨(dú)一種數(shù)學(xué)游戲,玩家需要根據(jù)9

答題星2018-06-03

您可以通過(guò)答題星輕松地創(chuàng)建試卷

全階人生考試2018-06-03

各種考試題,題庫(kù),初中,高中,大學(xué)四六

運(yùn)動(dòng)步數(shù)有氧達(dá)人2018-06-03

記錄運(yùn)動(dòng)步數(shù),積累氧氣值。還可偷

每日養(yǎng)生app2018-06-03

每日養(yǎng)生,天天健康

體育訓(xùn)練成績(jī)?cè)u(píng)定2018-06-03

通用課目體育訓(xùn)練成績(jī)?cè)u(píng)定