1. 简介
当我们在使用Springboot进行开发时,可能会遇到一些异步任务的场景,如果每次执行这些异步任务都去新建一个异步线程来执行的话,那么代码就太冗余了。Springboot提供了@Async
注解,能够轻松地对这些异步任务进行执行
失效条件
异步方法使用
static
修饰调用方法和异步方法在同一个类中
2. 代码工程
基于springboot实现@Async异步任务
- 添加依赖
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<parent>
<artifactId>springboot-demo</artifactId>
<groupId>com.ygb</groupId>
<version>1.0-SNAPSHOT</version>
</parent>
<modelVersion>4.0.0</modelVersion>
<artifactId>async</artifactId>
<properties>
<maven.compiler.source>8</maven.compiler.source>
<maven.compiler.target>8</maven.compiler.target>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-autoconfigure</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
</dependency>
</dependencies>
</project>
Service
thenApply
: 处理上一阶段计算结果thenCompose
: 整合两个计算结果1
2
3
4
5
6
7
8
9
10
public class FirstAsyncService {
public CompletableFuture<String> asyncGetData() throws InterruptedException {
log.info("Execute method asynchronously " + Thread.currentThread().getName());
Thread.sleep(4000);
return new AsyncResult<>(super.getClass().getSimpleName() + " response !!! ").completable();
}
}1
2
3
4
5
6
7
8
9
10
11
public class SecondAsyncService {
public CompletableFuture<String> asyncGetData() throws InterruptedException {
log.info("Execute method asynchronously " + Thread.currentThread()
.getName());
Thread.sleep(4000);
return new AsyncResult<>(super.getClass().getSimpleName() + " response !!! ").completable();
}
}整合两个计算结果
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
public class NotifyService {
public void noAsync() {
log.info("Execute method asynchronously. " + Thread.currentThread().getName());
}
public void withAsync() {
log.info("Execute method asynchronously. " + Thread.currentThread().getName());
}
public void mockerror() {
int ss=12/0;
}
public Future<String> asyncMethodWithReturnType() {
log.info("Execute method asynchronously - " + Thread.currentThread().getName());
try {
Thread.sleep(5000);
return new AsyncResult<String>("hello world !!!!");
} catch (InterruptedException e) {
e.printStackTrace();
}
return null;
}
private FirstAsyncService fisrtService;
private SecondAsyncService secondService;
public CompletableFuture<String> asyncMergeServicesResponse() throws InterruptedException {
CompletableFuture<String> fisrtServiceResponse = fisrtService.asyncGetData();
CompletableFuture<String> secondServiceResponse = secondService.asyncGetData();
// Merge responses from FirstAsyncService and SecondAsyncService
return fisrtServiceResponse.thenCompose(fisrtServiceValue -> secondServiceResponse.thenApply(secondServiceValue -> fisrtServiceValue + secondServiceValue));
}
}
config,使用
@EnableAsync
开启1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public class SpringAsyncConfig implements AsyncConfigurer {
public Executor threadPoolTaskExecutor() {
return new ThreadPoolTaskExecutor();
}
public Executor getAsyncExecutor() {
ThreadPoolTaskExecutor threadPoolTaskExecutor = new ThreadPoolTaskExecutor();
threadPoolTaskExecutor.initialize();
return threadPoolTaskExecutor;
}
public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
return new CustomAsyncExceptionHandler();
}
}自定义异常处理
1
2
3
4
5
6
7
8
9
10
11
12
13
14public class CustomAsyncExceptionHandler implements AsyncUncaughtExceptionHandler {
public void handleUncaughtException(
Throwable throwable, Method method, Object... obj) {
System.out.println("Exception message - " + throwable.getMessage());
System.out.println("Method name - " + method.getName());
for (Object param : obj) {
System.out.println("Parameter value - " + param);
}
}
}启动类
1
2
3
4
5
6
7
public class DemoApplication {
public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}
}测试类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
public class DemoTests {
NotifyService notifyService;
public void before() {
log.info("init some data");
}
public void after(){
log.info("clean some data");
}
public void execute() throws ExecutionException, InterruptedException {
log.info("your method test Code");
log.info("Invoking an asynchronous method. " + Thread.currentThread().getName());
notifyService.noAsync();
notifyService.withAsync();
}
public void mockerror() throws ExecutionException, InterruptedException {
notifyService.mockerror();
}
public void testAsyncAnnotationForMethodsWithReturnType()
throws InterruptedException, ExecutionException {
log.info("Invoking an asynchronous method. " + Thread.currentThread().getName());
Future<String> future = notifyService.asyncMethodWithReturnType();
while (true) {
if (future.isDone()) {
log.info("Result from asynchronous process - " + future.get());
break;
}
log.info("Continue doing something else. ");
Thread.sleep(1000);
}
}
public void testAsyncAnnotationForMergedServicesResponse() throws InterruptedException, ExecutionException {
log.info("Invoking an asynchronous method. " + Thread.currentThread().getName());
CompletableFuture<String> completableFuture = notifyService.asyncMergeServicesResponse();
while (true) {
if (completableFuture.isDone()) {
log.info("Result from asynchronous process - " + completableFuture.get());
break;
}
log.info("Continue doing something else. ");
Thread.sleep(1000);
}
}
}1
2
3
4
5
62024-11-15 14:17:24.723 INFO 1066 --- [ main] com.et.async.DemoTests : init some data
2024-11-15 14:17:24.724 INFO 1066 --- [ main] com.et.async.DemoTests : your method test Code
2024-11-15 14:17:24.724 INFO 1066 --- [ main] com.et.async.DemoTests : Invoking an asynchronous method. main
2024-11-15 14:17:24.730 INFO 1066 --- [ main] com.et.async.service.NotifyService : Execute method asynchronously. main
2024-11-15 14:17:24.730 INFO 1066 --- [ main] com.et.async.service.NotifyService : Execute method asynchronously. main
2024-11-15 14:17:24.732 INFO 1066 --- [ main] com.et.async.DemoTests : clean some data