⭐⭐⭐ Spring Boot 项目实战 ⭐⭐⭐ Spring Cloud 项目实战
《Dubbo 实现原理与源码解析 —— 精品合集》 《Netty 实现原理与源码解析 —— 精品合集》
《Spring 实现原理与源码解析 —— 精品合集》 《MyBatis 实现原理与源码解析 —— 精品合集》
《Spring MVC 实现原理与源码解析 —— 精品合集》 《数据库实体设计合集》
《Spring Boot 实现原理与源码解析 —— 精品合集》 《Java 面试题 + Java 学习指南》

摘要: 原创出处 blog.csdn.net/weixin_43356538/article/details/129705507 「Tiny青年」欢迎转载,保留摘要,谢谢!


🙂🙂🙂关注**微信公众号:【芋道源码】**有福利:

  1. RocketMQ / MyCAT / Sharding-JDBC 所有源码分析文章列表
  2. RocketMQ / MyCAT / Sharding-JDBC 中文注释源码 GitHub 地址
  3. 您对于源码的疑问每条留言将得到认真回复。甚至不知道如何读源码也可以请教噢
  4. 新的源码解析文章实时收到通知。每周更新一篇左右
  5. 认真的源码交流微信群。

自定义线程池

@Configuration
public class ThreadPoolConfig {

public static ThreadPoolExecutor getThreadPoolExecutor() {
int availableProcessors = Runtime.getRuntime().availableProcessors();
return new ThreadPoolExecutor(
availableProcessors,
availableProcessors,
0L,
TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<>(9999),
new ThreadFactoryBuilder().setNameFormat("custom-thread-pool-%d").build(),
new ThreadPoolExecutor.CallerRunsPolicy());
}

}

程序存在异常,却返回成功

写一个存在异常的程序,让其异步执行

public static final ThreadPoolExecutor CUSTOM_THREAD_POOL = ThreadPoolConfig.getThreadPoolExecutor();

/**
* 异步执行异常测试
*/
@ApiOperation(value = "异步执行异常测试", code = 800)
@GetMapping("/asyncException")
public ResponseData<Object> asyncException() {
try {
try {
CompletableFuture.runAsync(() -> {
int i = 1 / 0;
}, CUSTOM_THREAD_POOL);
} catch (Exception e) {
log.error("异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}
return new ResponseData<>(StatusCodeEnum.SUCCESS_CODE.getStatusCode(), "操作成功");
} catch (Exception e) {
return new ResponseData<>(StatusCodeEnum.ERROR_CODE.getStatusCode(), "操作失败:" + e.getMessage());
}
}

结果:接口返回成功,控制台没有打印错误信息。

异步调用join()

// join方法获取异常信息: 将异步线程中发生的异常信息抛到主线程, 这样异常可被主线程捕获
try {
CompletableFuture.runAsync(() -> {
int i = 1 / 0;
}, CUSTOM_THREAD_POOL).join();
} catch (Exception e) {
log.error("外层异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}

结果:接口返回失败,控制台打印异常日志。

异步调用get()

异步方法中get()是阻塞的,在使用时要设置超时时间。

// get方法获取异常信息: 将异步线程中发生的异常信息抛到主线程, 这样异常可被主线程捕获
try {
CompletableFuture.runAsync(() -> {
int i = 1 / 0;
}, CUSTOM_THREAD_POOL).get(2, TimeUnit.SECONDS);
} catch (Exception e) {
log.error("外层异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}

结果:接口返回成功,控制台打印异常信息。

异步调用exception()

// exceptionally获取异常信息: 异常是存在于异步当中的, 不能被主线程捕获
try {
CompletableFuture.runAsync(() -> {
int i = 1 / 0;
}, CUSTOM_THREAD_POOL)
.exceptionally(e -> {
log.error("异步运行异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
});
} catch (Exception e) {
log.error("异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}

结果:接口返回成功,控制台打印异步线程异常日志,主线程没有打印异常日志

异步调用whenComplete()

// whenComplete获取异常信息: 异常是存在于异步当中的, 不能被主线程捕获
try {
CompletableFuture.runAsync(() -> {
int i = 1 / 0;
}, CUSTOM_THREAD_POOL)
.whenComplete((r, e) -> {
if (e != null) {
log.error("异步执行异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}
});
} catch (Exception e) {
log.error("异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}

结果:结果返回成功,控制台打印异步线程异常信息,主线程没有打印异常信息

异步调用handle()

// handle获取异常信息: 异常是存在于异步当中的, 不能被主线程捕获
try {
CompletableFuture.runAsync(() -> {
int i = 1 / 0;
}, CUSTOM_THREAD_POOL)
.handle((r, e) -> {
if (e != null) {
log.error("异步执行异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}
return null;
});
} catch (Exception e) {
log.error("异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}

结果:结果返回成功,控制台打印异步线程异常信息,主线程没有打印异常信息

程序发生异常时需要做处理,可以调用get()/join()

try {
CompletableFuture.runAsync(() -> {
int i = 1 / 0;
}, CUSTOM_THREAD_POOL)
.exceptionally(e -> {
log.error("异步执行异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}).join();
} catch (Exception e) {
log.error("异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}

try {
CompletableFuture.runAsync(() -> {
int i = 1 / 0;
}, CUSTOM_THREAD_POOL)
.exceptionally(e -> {
log.error("异步执行异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}).get(2, TimeUnit.SECONDS);
} catch (Exception e) {
log.error("异常信息: " + e.getMessage(), e);
throw new BusinessException(e.getMessage());
}

程序发生异常时不做处理直接报错,直接调用get()/join()

直接的异步方法后调用get()/join()。

总结

在使用异步CompletableFuture时,无论是否有返回值都要调用get()/join()方法,避免程序执行报错了,仍然返回成功。如果在程序报错时需要对上一个异步任务结果做其他操作,可以调用whenComplete()handle()处理,如果只是对异常做处理,不涉及对上一个异步任务结果的情况,调用exceptionally()处理。

文章目录
  1. 1. 自定义线程池
  2. 2. 程序存在异常,却返回成功
  3. 3. 异步调用join()
  4. 4. 异步调用get()
  5. 5. 异步调用exception()
  6. 6. 异步调用whenComplete()
  7. 7. 异步调用handle()
  8. 8. 程序发生异常时需要做处理,可以调用get()/join()
  9. 9. 程序发生异常时不做处理直接报错,直接调用get()/join()
  10. 10. 总结