成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

強大的異步任務處理類CompletableFuture使用詳解

開發 前端
Future是從JDK1.5開始有的,目的是獲取異步任務執行的結果,通常情況會結合ExecutorService及Callable一起使用。

環境:Java8

在Java 8中, 新增加了一個CompletableFuture類,該類提供了差不多50個左右的方法(都是用來完成各種異步場景需求),并且結合了Future的優點(繼承自Future類),提供了比Future更為強大的功能,這使得在異步編程方面變的簡單,同時還提供了函數式編程的能力,可以通過回調的方式處理計算結果,并且提供了轉換和組合CompletableFuture的各種方法。

Future基本應用

Future是從JDK1.5開始有的,目的是獲取異步任務執行的結果,通常情況會結合ExecutorService及Callable一起使用。

1. Future結合Callable使用

單任務執行

private static class Task implements Callable<String> {


  @Override
  public String call() throws Exception {
    TimeUnit.SECONDS.sleep(3) ;
    return "success";
  }


}
public static void main(String[] args) throws Exception {
  ThreadPoolExecutor executor = new ThreadPoolExecutor(3, 3, 60, TimeUnit.SECONDS, new ArrayBlockingQueue<>(10)) ;
  Future<String> future = executor.submit(new Task()) ;
  String result = future.get() ;
  System.out.println("執行結果:" + result) ;
}

當執行到future.get()方法的時候會阻塞,等待3s后繼續執行。

多個任務同時執行

private static class Task implements Callable<String> {
  private int sleep ;
  public Task(int sleep) {
    this.sleep = sleep ;
  }


  @Override
  public String call() throws Exception {
    TimeUnit.SECONDS.sleep(this.sleep) ;
    return "success";
  }
}
public static void main(String[] args) throws Exception {
  ThreadPoolExecutor executor = new ThreadPoolExecutor(3, 3, 60, TimeUnit.SECONDS, new ArrayBlockingQueue<>(10)) ;
  Future<String> future1 = executor.submit(new Task(3)) ;
  Future<String> future2 = executor.submit(new Task(2)) ;
  Future<String> future3 = executor.submit(new Task(1)) ;
  String result1 = future1.get() ;
  String result2 = future2.get() ;
  String result3 = future3.get() ;
  System.out.println("result1:" + result1 + "\t" + "result2:" + result2 + "\t" + "result3:" + result3) ;
}

以上代碼執行的3個任務分別用時3,2,1s。future1用時最長。

從運行的結果看到即便future2, future3執行時間短也必須等待future1執行完后才會繼續,雖然你可以倒過來獲取結果,但是在實際項目中的應用你應該是不能確認每個任務執行需要多長時間,誰先執行完就先獲取誰。

雖然這種同步阻塞的方式在有些場景下還是很有必要的。但由于它的同步阻塞導致了當前線程不能干其它的事必須一致等待。

CompletionService解決Future的缺點

CompletionService是一邊生產新的任務,一邊處理已經完成的任務。簡單地說就是CompletionService不管任務執行先后順序,誰先執行完就處理誰。

private static class Task implements Callable<String> {
  private int time;
  private String name ;
  public Task(int time, String name) {
    this.time = time ;
    this.name = name ;
  }
  @Override
  public String call() throws Exception {
    TimeUnit.SECONDS.sleep(this.time) ;
    return name ;
  }


}
public static void main(String[] args) throws Exception {
  ThreadPoolExecutor pool = new ThreadPoolExecutor(3, 3, 60, TimeUnit.SECONDS, new ArrayBlockingQueue<>(10)) ;
  CompletionService<String> cs = new ExecutorCompletionService<>(pool) ;
  cs.submit(new Task(3, "name" + 3)) ;
  cs.submit(new Task(1, "name" + 1)) ;
  cs.submit(new Task(2, "name" + 2)) ;
  for (int i = 0; i < 3; i++) {
    System.out.println(cs.take().get()) ;
  }
}

通過執行結果發現,任務的結果獲取是以誰先執行完處理誰與任務的執行先后沒有關系。

2. CompletableFuture異步編程

CompletableFuture通過如下4個靜態方法來執行異步任務

圖片圖片

2.1 簡單異步任務鏈式調用執行

ThreadPoolExecutor executor = new ThreadPoolExecutor(5, 5, 60, TimeUnit.SECONDS, new ArrayBlockingQueue<>(10)) ;
CompletableFuture.runAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(3) ;
    System.out.println(Thread.currentThread().getName() + ", 1 任務執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
}, executor).thenRun(() -> {
  try {
    TimeUnit.SECONDS.sleep(2) ;
    System.out.println(Thread.currentThread().getName() + ", 2 任務執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
}) ;
System.out.println("主線程:" + Thread.currentThread().getName()) ;
executor.shutdown() ;

執行結果:

圖片圖片

2.2 獲取上一步任務執行結果及任務完成處理

CompletableFuture.supplyAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(3) ;
    System.out.println(Thread.currentThread().getName() + ", 1 任務執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return "1" ;
}, executor).thenApply(res -> {
  System.out.println("獲取到上一步任務執行結果:" + res) ;
  try {
    TimeUnit.SECONDS.sleep(2) ;
    System.out.println(Thread.currentThread().getName() + ", 2 任務執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return "2" ;
}).whenComplete((res, tx) -> {
  System.out.println("獲取到結果:" + res) ;
  if (tx != null) {
    System.err.println("發生錯誤了:" + tx.getMessage()) ;
  }
  executor.shutdown();
}) ;
System.out.println("主線程:" + Thread.currentThread().getName()) ;

執行結果:

圖片圖片

這里如果任務執行的時候發生了異常那么在whenComplete方法中的res 會為空,tx為發生異常的對象。沒有異常時res有執行的機構,tx異常對象為空。

2.3 異步任務異常處理

CompletableFuture.supplyAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(3) ;
    System.out.println(Thread.currentThread().getName() + ", 1 任務執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return "1" ;
}, executor).thenApply(res -> {
  System.out.println("獲取到上一步任務執行結果:" + res) ;
  try {
    TimeUnit.SECONDS.sleep(2) ;
    System.out.println(Thread.currentThread().getName() + ", 2 任務執行完成") ;
    System.out.println(1 / 0) ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return "2" ;
}).exceptionally(tx -> {
  System.out.println(Thread.currentThread().getName() + ", 任務執行發生了異常") ;
  return "error" ;
}).whenComplete((res, tx) -> {
  System.out.println("獲取到結果:" + res) ;
  if (tx != null) {
    System.err.println("發生錯誤了:" + tx.getMessage()) ;
  }
  executor.shutdown();
}) ;
System.out.println("主線程:" + Thread.currentThread().getName()) ;

這里我們人為的制造異常 1 / 0 。

執行結果:

圖片圖片

根據執行結果當發生異常時進入exceptionally方法,最終進入whenComplete方法此時 tx異常對象是發生異常的異常對象。

2.4 所有任務完成才算完成任務

CompletableFuture.allOf

CompletableFuture<Double> calc1 = CompletableFuture.supplyAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(2) ;
    System.out.println(Thread.currentThread().getName() + ", calc1任務執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return 10D ;
}, executor) ;


CompletableFuture<Double> calc2 = CompletableFuture.supplyAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(5) ;
    System.out.println(Thread.currentThread().getName() + ", calc2任務執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return 20D ;
}, executor) ;
// 當任何一個任務發生異常,這里的tx都不會為null
CompletableFuture.allOf(calc1, calc2).whenComplete((res, tx) -> {
  System.out.println("獲取到結果:" + res + ", " + tx) ;
  try {
    System.out.println(calc1.get()) ;
    System.out.println(calc2.get()) ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  } catch (ExecutionException e) {
    e.printStackTrace();
  }
  executor.shutdown();
}) ;

執行結果:

圖片圖片

在這里whenComplete中的res是沒有結果的,要獲取數據我們的分別調用get方法獲取。

2.5 handle方法對結果處理

CompletableFuture.supplyAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(2) ;
    System.out.println(Thread.currentThread().getName() + ", 1 任務執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return "0" ;
}, executor).handle((res, tx) -> {
  // 處理結果數據
  return res + "1" ;
}).whenComplete((res, tx) -> {
  System.out.println("獲取到結果:" + res) ;
  if (tx != null) {
    System.err.println("發生錯誤了:" + tx.getMessage()) ;
  }
  executor.shutdown();
}) ;

執行結果:

正確

圖片圖片

發生異常時:

圖片

當發生異常時handle方法中的res是沒有值的,tx異常對象為發生異常的異常對象。

2.6 合并異步任務

將兩個異步任務完成后合并處理

CompletableFuture.thenCombine

CompletableFuture<Double> task1 = CompletableFuture.supplyAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(2) ;
    System.out.println(Thread.currentThread().getName() + ", 任務1執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return 10d ;
}, executor) ;
CompletableFuture<Double> task2 = CompletableFuture.supplyAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(2) ;
    System.out.println(Thread.currentThread().getName() + ", 任務2執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return 20d ;
}, executor) ;
task1.thenCombine(task2, (t1, t2) -> {
  System.out.println(Thread.currentThread().getName() + ", 合并任務完成") ;
  return t1 + "," + t2 ;
}).whenComplete((res, tx) -> {
  System.out.println("獲取到結果:" + res) ;
  if (tx != null) {
    System.err.println("發生錯誤了:" + tx.getMessage()) ;
  }
  executor.shutdown();
}) ;

執行結果:

圖片圖片

2.7 異步任務誰快誰就進入下一步的執行

CompletableFuture.applyToEither

兩個異步任務誰先執行完誰就繼續執行后續的操作。

CompletableFuture<Double> task1 = CompletableFuture.supplyAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(2) ;
    System.out.println(Thread.currentThread().getName() + ", 任務1執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return 10d ;
}, executor) ;
CompletableFuture<Double> task2 = CompletableFuture.supplyAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(2) ;
    System.out.println(Thread.currentThread().getName() + ", 任務2執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return 20d ;
}, executor) ;
task1.applyToEither(task2, res -> {
  return res ;
}).whenComplete((res, tx) -> {
  System.out.println("獲取到結果:" + res) ;
  if (tx != null) {
    System.err.println("發生錯誤了:" + tx.getMessage()) ;
  }
  executor.shutdown();
}) ;

執行結果:

圖片圖片

2.8 兩個異步任務都執行完了才繼續執行

只有兩個任務都執行完成了后才會繼續。

CompletableFuture.runAfterBoth

CompletableFuture<Double> task1 = CompletableFuture.supplyAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(2) ;
    System.out.println(Thread.currentThread().getName() + ", 任務1執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return 10d ;
}, executor) ;
CompletableFuture<Double> task2 = CompletableFuture.supplyAsync(() -> {
  try {
    TimeUnit.SECONDS.sleep(2) ;
    System.out.println(Thread.currentThread().getName() + ", 任務2執行完成") ;
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
  return 20d ;
}, executor) ;
task1.runAfterBoth(task2, () -> {
  System.out.println("任務都執行完成了...") ;
}).whenComplete((res, tx) -> {
  System.out.println("獲取到結果:" + res) ;
  if (tx != null) {
    System.err.println("發生錯誤了:" + tx.getMessage()) ;
  }
  executor.shutdown();
}) ;

執行結果:

圖片圖片

2.9 任意一個任務執行完成就算完成

CompletableFuture.anyOf

CompletableFuture<String> task1 = CompletableFuture.supplyAsync(() -> {
  sleep(1000) ;
  System.out.println("我是任務1") ;
  return "Task1" ;
}, executor) ;


CompletableFuture<String> task2 = CompletableFuture.supplyAsync(() -> {
  sleep(3000) ;
  System.out.println("我是任務2") ;
  System.out.println(1 / 0) ;
  return "Task2" ;
}, executor) ;
// 任意一個任務執行完成就算完成
// 當任務執行發生異常后,th才不會為null
CompletableFuture.anyOf(task1, task2).whenCompleteAsync((v, th) -> {
  System.out.println("v = " + v) ;
  System.out.println("th = " + th) ;
}, executor) ;

執行結果:

圖片圖片

2.10 接收上一個任務的執行結果

CompletableFuture.supplyAsync(() -> {
  sleep(2000) ;
  System.out.println("第一個任務執行完成...") ;
  // System.out.println(1 / 0) ;
  return new Random().nextInt(10000) ;
}, executor).thenAcceptAsync(res -> { // 接收上一個任務的執行結果
  System.out.println("任務執行結果:" + res) ;
}, executor) ;

執行結果:

圖片圖片

以上是本篇文章的全部內容,希望對你有幫助。

責任編輯:武曉燕 來源: Spring全家桶實戰案例源碼
相關推薦

2015-06-16 11:06:42

JavaCompletable

2021-06-06 16:56:49

異步編程Completable

2017-12-21 15:48:11

JavaCompletable

2024-10-14 13:12:59

2025-02-28 09:20:00

Future開發代碼

2024-10-14 08:29:14

異步編程任務

2023-01-03 10:38:04

函數計算技術

2024-01-11 12:14:31

Async線程池任務

2024-12-26 12:59:39

2024-04-18 08:20:27

Java 8編程工具

2014-12-02 10:02:21

Android異步任務

2021-02-21 14:35:29

Java 8異步編程

2024-08-06 09:43:54

Java 8工具編程

2010-12-01 14:34:59

AsyncTask異步處理任務Android

2022-07-08 14:14:04

并發編程異步編程

2025-04-30 01:50:00

C#異步編程

2022-04-19 09:03:22

Linuxcron命令

2022-05-13 12:34:16

美團開發實踐

2022-09-16 11:23:59

Python框架Celery

2020-05-29 07:20:00

Java8異步編程源碼解讀
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 色站综合 | 久久久久久久综合 | 日本不卡在线视频 | 一区二区免费在线 | 国产精品二区三区在线观看 | 天天干天天爱天天操 | 亚洲精品国产a久久久久久 中文字幕一区二区三区四区五区 | 91在线网| 97在线观视频免费观看 | 日韩中文字幕一区二区 | 九九热这里 | 欧美一级久久久猛烈a大片 日韩av免费在线观看 | 国产日韩一区二区 | 日韩成人免费av | 欧美激情第一区 | 日韩精品一区二区三区中文在线 | www.99热| 国产美女在线免费观看 | 国产精品大片在线观看 | 欧美精品三区 | 国产精品久久久久久久久久妞妞 | 久久99精品久久久久 | 色婷婷av一区二区三区软件 | 成人二区| 欧美精品一区二区三区四区 在线 | 91免费看片 | 中文字幕在线三区 | 欧美日韩电影一区二区 | 欧美视频二区 | 一级黄色影片在线观看 | 国产精品久久久久久久午夜片 | 午夜视频在线免费观看 | 四虎精品在线 | 91精品国产91久久久久游泳池 | 久久欧美高清二区三区 | 久久99精品久久久久久噜噜 | 中文字幕视频免费 | av网站免费在线观看 | 一区二区三区av夏目彩春 | 美女视频黄色的 | 国产成人高清视频 |