偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Java異步編程七種實(shí)現(xiàn)方法,最后一種非常強(qiáng)大

開(kāi)發(fā) 前端
異步編程是一種編程范式,旨在提高程序的響應(yīng)性和性能,尤其在處理耗時(shí)操作時(shí)顯得尤為重要。它通過(guò)允許程序在等待某些操作(如I/O操作、網(wǎng)絡(luò)請(qǐng)求或數(shù)據(jù)庫(kù)查詢(xún))完成時(shí)繼續(xù)執(zhí)行其他任務(wù),從而優(yōu)化資源利用。

環(huán)境:java21

1. 簡(jiǎn)介

異步編程是一種編程范式,旨在提高程序的響應(yīng)性和性能,尤其在處理耗時(shí)操作時(shí)顯得尤為重要。它通過(guò)允許程序在等待某些操作(如I/O操作、網(wǎng)絡(luò)請(qǐng)求或數(shù)據(jù)庫(kù)查詢(xún))完成時(shí)繼續(xù)執(zhí)行其他任務(wù),從而優(yōu)化資源利用。異步編程避免了傳統(tǒng)同步編程中的阻塞問(wèn)題,使得程序更加高效和流暢。異步編程現(xiàn)已成為處理并發(fā)和I/O密集型任務(wù)的重要手段。

本篇文章中我們將介紹在Java中實(shí)現(xiàn)異步編程的7種方法,這其中會(huì)涉及到幾個(gè)非常優(yōu)秀的第三方類(lèi)庫(kù)。

2. 異步編程

2.1 Thread

Thread類(lèi)是Java中用于創(chuàng)建和管理線(xiàn)程的核心類(lèi)。通過(guò)繼承Thread類(lèi)或?qū)崿F(xiàn)Runnable接口,可以創(chuàng)建新的線(xiàn)程。Thread類(lèi)提供了線(xiàn)程的啟動(dòng)、運(yùn)行、中斷、優(yōu)先級(jí)設(shè)置等方法,是實(shí)現(xiàn)多線(xiàn)程編程的基礎(chǔ)。

隨著Java 8中引入lambda表達(dá)式,代碼變得更加清晰和易讀,如下示例,在一個(gè)新線(xiàn)程中打印一個(gè)數(shù)的階乘。

public class FactorialThread extends Thread {
  private int number;
  public FactorialThread(int number, String name) {
    super(name) ;
    this.number = number;
  }
  // 計(jì)算階乘的方法
  private long factorial(int n) {
    long result = 1;
    for (int i = 1; i <= n; i++) {
      result *= i;
    }
    return result;
  }
  @Override
  public void run() {
    long result = factorial(number);
    System.out.println(Thread.currentThread().getName() + " -> " + number + " 的階乘是: " + result);
  }
  public static void main(String[] args) {
    // 創(chuàng)建并啟動(dòng)線(xiàn)程,計(jì)算并打印5的階乘
    Thread t1 = new FactorialThread(5, "T1");
    t1.start();
    // 可以創(chuàng)建更多線(xiàn)程來(lái)計(jì)算其他數(shù)的階乘
    Thread t2 = new FactorialThread(7, "T2");
    t2.start();
  }
}

輸出結(jié)果

T1 -> 5 的階乘是: 120
T2 -> 7 的階乘是: 5040

我們還可以直接通過(guò)lambda計(jì)算:

Thread t3 = new Thread(() -> {
  int num = 8 ;
  int result = 1 ;
  for (int i = 1; i <= num; i++) {
    result *= i ;
  }
  System.out.println(Thread.currentThread().getName() + " -> " + num + " 的階乘是: " + result);
}, "T3") ;
t3.start() ;

通過(guò)lambda使得程序更加的簡(jiǎn)潔易懂。

2.2 FutureTask

從Java 5起,F(xiàn)uture接口就提供了一種通過(guò)FutureTask執(zhí)行異步操作的方式。我們可以利用ExecutorService的submit方法來(lái)異步執(zhí)行任務(wù),并返回FutureTask的實(shí)例。下面通過(guò)FutureTask計(jì)算階乘。

try (ThreadPoolExecutor executor = new ThreadPoolExecutor(5, 5, 
    60, TimeUnit.SECONDS, new ArrayBlockingQueue<>(10))) {
  int num = 8 ;
  // 返回FutureTask實(shí)例
  Future<Integer> task = executor.submit(() -> {
    int result = 1 ;
    for (int i = 1; i <= num; i++) {
      result *= i ;
    }
    return result ;
  }) ;
  // 獲取執(zhí)行結(jié)果
  Integer result = task.get() ;
  System.out.printf("%s -> %s 的階乘是: %s%n", Thread.currentThread().getName(), num, result);
  executor.shutdown() ;
}

Future還提供了isDone方法,我們可以調(diào)用該方法判斷當(dāng)前是否執(zhí)行完成。

if (!task.isDone()) {
  System.err.println("還未執(zhí)行完成") ;
}

上面的get方法調(diào)用會(huì)將當(dāng)前線(xiàn)程阻塞住,直到返回結(jié)果或者拋出異常。

Future<Integer> task = executor.submit(() -> {
  // ...
  System.out.println(1 / 0) ;
  return result ;
}) ;

若將程序修改為上面,如果我們沒(méi)有調(diào)用Future#get方法,那么我們將無(wú)法得知程序拋出了異常。

2.3 CompletableFuture

CompletableFuture是Java 8引入的一個(gè)類(lèi),它結(jié)合了FutureCompletionStage的特點(diǎn),提供了強(qiáng)大的異步編程能力。通過(guò)豐富的API,它可以異步執(zhí)行任務(wù)、處理結(jié)果、組合多個(gè)異步操作,并支持異常處理,極大地簡(jiǎn)化了異步編程的復(fù)雜性。下面通過(guò)該類(lèi)實(shí)現(xiàn)階乘計(jì)算。

final int num = 10 ;
CompletableFuture<Long> cf = CompletableFuture.supplyAsync(() -> {
  long result = 1 ;
  for (int i = 1; i <= num; i++) {
    result *= i ;
  }
  return result ;
}) ;
Long result = cf.get() ;
System.out.printf("%s -> %s 的階乘是: %s%n", Thread.currentThread().getName(), num, result);

我們不需要顯式地使用ExecutorServiceCompletableFuture內(nèi)部使用ForkJoinPool來(lái)異步處理任務(wù)。因此,它使我們的代碼更加簡(jiǎn)潔。

我們還可以通過(guò)調(diào)用join方法來(lái)等待異步線(xiàn)程的執(zhí)行完成。

CompletableFuture.runAsync(() -> {
  System.out.printf("%s - %s 開(kāi)始執(zhí)行任務(wù)%n", Thread.currentThread().getName(), System.currentTimeMillis()) ;
  try {
    // 模擬耗時(shí)任務(wù)
    TimeUnit.SECONDS.sleep(2) ;
  } catch (InterruptedException e) {}
  System.out.printf("%s - %s 任務(wù)執(zhí)行完成%n", Thread.currentThread().getName(), System.currentTimeMillis()) ;
}).join() ;

輸出結(jié)果

ForkJoinPool.commonPool-worker-1 - 1738806351732 開(kāi)始執(zhí)行任務(wù)
ForkJoinPool.commonPool-worker-1 - 1738806353742 任務(wù)執(zhí)行完成

2.4 Guava

Guava提供了ListenableFuture類(lèi)來(lái)執(zhí)行異步操作,允許注冊(cè)回調(diào)函數(shù)以處理操作完成時(shí)的結(jié)果或異常,增強(qiáng)了Future的功能。

引入依賴(lài)

<dependency>
  <groupId>com.google.guava</groupId>
  <artifactId>guava</artifactId>
  <version>33.4.0-jre</version>
</dependency>

還是以計(jì)算階乘為例:

ThreadPoolExecutor executor = new ThreadPoolExecutor(5, 5, 
  60, TimeUnit.SECONDS, new ArrayBlockingQueue<>(10)) ;
ListeningExecutorService les = MoreExecutors.listeningDecorator(executor);
int num = 8 ;
ListenableFuture<Long> future = les.submit(() -> {
  long result = 1 ;
  for (int i = 1; i <= num; i++) {
    result *= i ;
  }
  return result ;
}) ;
System.out.printf("%s -> %s 的階乘是: %s%n", Thread.currentThread().getName(), num, future.get());

我們還可以通過(guò)注冊(cè)回調(diào)的機(jī)制來(lái)獲取結(jié)果

future.addListener(() -> {
  System.err.printf("%s - 計(jì)算完成%n", Thread.currentThread().getName()) ;
  try {
    System.out.printf("%s -> %s 的階乘是: %s%n", Thread.currentThread().getName(), num, future.get());
  } catch (InterruptedException | ExecutionException e) {
  }
}, executor) ;

輸出結(jié)果

pool-1-thread-2 - 計(jì)算完成
pool-1-thread-2 -> 8 的階乘是: 40320

當(dāng)Future執(zhí)行完成后,會(huì)自動(dòng)執(zhí)行注冊(cè)的監(jiān)聽(tīng)器,你也可以注冊(cè)多個(gè)監(jiān)聽(tīng)器(但是不保證執(zhí)行的順序)。

2.5 EA Async

ea-async庫(kù)允許以順序方式編寫(xiě)異步(非阻塞)代碼。因此,它使得異步編程變得更加容易,并且能夠自然地?cái)U(kuò)展。

引入依賴(lài)

<dependency>
  <groupId>com.ea.async</groupId>
  <artifactId>ea-async</artifactId>
  <version>1.2.3</version>
</dependency>

注:多年未更新了。

static {
  Async.init(); 
}
public static long factorial(long num) {
  long result = 1 ;
  for (int i = 1; i <= num; i++) {
    result *= i ;
  }
  return result ;
}
public static void main(String[] args) {
  final long num = 10 ;
  CompletableFuture<Long> completableFuture = CompletableFuture.supplyAsync(() -> factorial(num));
  long result = Async.await(completableFuture) ;
  System.out.printf("%s -> %s 的階乘是: %s%n", Thread.currentThread().getName(), num, result) ;
}

Async#await這個(gè)方法的行為類(lèi)似于 CompletableFuture.join(),但實(shí)際上它會(huì)使調(diào)用者返回一個(gè)承諾(promise)而不是阻塞。

注:當(dāng)前必須在JDK11中運(yùn)行。

2.6 Cactoos

Cactoos 是一組面向?qū)ο蟮?Java 基本元素集合。cactoos出現(xiàn)的動(dòng)機(jī):對(duì) JDK、Guava 和 Apache Commons 并不滿(mǎn)意,因?yàn)樗鼈兪沁^(guò)程式的,而不是面向?qū)ο蟮?。它們能完成工作,但主要是通過(guò)靜態(tài)方法來(lái)實(shí)現(xiàn)的。Cactoos 建議幾乎做同樣的事情,但要通過(guò)對(duì)象來(lái)實(shí)現(xiàn)。

引入依賴(lài)

<dependency>
  <groupId>org.cactoos</groupId>
  <artifactId>cactoos</artifactId>
  <version>0.56.1</version>
</dependency>

使用該庫(kù)計(jì)算階乘:

public static long factorial(long num) {
  long result = 1 ;
  for (int i = 1; i <= num; i++) {
    result *= i ;
  }
  System.err.println("當(dāng)前執(zhí)行線(xiàn)程: " + Thread.currentThread().getName()) ;
  return result ;
}
public static void main(String[] args) throws Exception {
  final long num = 10 ;
  Async<Long, Long> asyncFunction = new Async<>(input -> factorial(input)) ;
  Future<Long> asyncFuture = asyncFunction.apply(num);
  long result = asyncFuture.get() ;
  System.out.printf("%s -> %s 的階乘是: %s%n", Thread.currentThread().getName(), num, result);
}

輸出結(jié)果:

當(dāng)前執(zhí)行線(xiàn)程: pool-1-thread-1
main -> 10 的階乘是: 3628800

Async構(gòu)造函數(shù)中,默認(rèn)會(huì)創(chuàng)建Executors.newSingleThreadExecutor線(xiàn)程池。

Cactoos 不僅僅是進(jìn)行異步編程的庫(kù),它還提供了其它非常多的功能,如下示例:

讀取文件

String text = new TextOf(
  new File("d:\\pack.txt")
).asString();

文本格式化

String name = "Spring Boot實(shí)戰(zhàn)案例100例" ;
String content = new FormattedText(
    "如何快速提升Spring技能, 必須學(xué)習(xí):%s",
    name
  ).asString() ;

詳細(xì)更多的示例查看如下鏈接:https://github.com/yegor256/cactoos

2.7 Jcabi-Aspects

Jcabi-Aspects 提供了 @Async 注解,通過(guò) AspectJ AOP 切面實(shí)現(xiàn)異步編程。

引入依賴(lài)

<dependency>
  <groupId>com.jcabi</groupId>
  <artifactId>jcabi-aspects</artifactId>
  <version>0.26.0</version>
</dependency>
<dependency>
  <groupId>org.aspectj</groupId>
  <artifactId>aspectjrt</artifactId>
  <version>1.9.20.1</version>
</dependency>

配置編譯插件

<plugin>
  <groupId>com.jcabi</groupId>
  <artifactId>jcabi-maven-plugin</artifactId>
  <version>0.17.0</version>
  <executions>
    <execution>
      <goals>
        <goal>ajc</goal>
      </goals>
    </execution>
  </executions>
</plugin>

計(jì)算階乘示例:

@Async
public void task(long num) {
  long result = 1 ;
  for (int i = 1; i <= num; i++) {
    result *= i ;
  }
  System.err.printf("%s - 計(jì)算結(jié)果: %s%n" , Thread.currentThread().getName(), result) ;
}
public static void main(String[] args) throws Exception {
  JcabiAspectTest at = new JcabiAspectTest() ;
  at.task(10L) ;
  System.in.read() ;
}

輸出結(jié)果:

jcabi-async - 計(jì)算結(jié)果: 3628800

配置了jcabi-maven-plugin插件后,在編譯階段會(huì)修改對(duì)應(yīng)的字節(jié)碼進(jìn)行代碼的增強(qiáng)。

jcabi-aspects庫(kù)不止異步功能,它還提供了其它非常強(qiáng)大的功能。如下的日志記錄功能:

@Async
@Loggable
public void task(long num) {
  // ...
}

當(dāng)task方法執(zhí)行時(shí),根據(jù)你當(dāng)前日志配置的級(jí)別會(huì)輸出相應(yīng)的日志信息,如下:

圖片圖片


最后:強(qiáng)烈推薦jcabi庫(kù),功能多又強(qiáng)大。

責(zé)任編輯:武曉燕 來(lái)源: Spring全家桶實(shí)戰(zhàn)案例源碼
相關(guān)推薦

2022-07-01 08:00:44

異步編程FutureTask

2024-10-21 13:05:40

2024-12-05 10:37:36

Java純函數(shù)final

2011-12-16 14:45:36

JavaJSP

2025-01-15 10:46:23

開(kāi)發(fā)JavaScript集合

2011-05-30 13:37:46

JSP

2010-09-16 17:47:49

2016-09-28 20:05:22

2010-10-15 10:02:01

Mysql表類(lèi)型

2025-05-13 08:20:58

2014-05-13 09:56:24

數(shù)據(jù)挖掘

2024-01-02 07:56:13

ReactuseEffect數(shù)據(jù)驅(qū)動(dòng) UI

2022-05-24 14:37:49

React條件渲染

2013-01-07 10:14:06

JavaJava枚舉

2009-07-23 16:50:04

2022-10-27 08:09:33

2023-04-18 15:57:30

2024-07-29 08:00:00

2011-03-14 10:46:03

2010-06-08 09:49:45

UML元件
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)