一、前言
❝
异步执行对于开发者来说并不陌生,在实际的开发过程中,很多场景多会使用到异步,相比同步执行,异步可以大大缩短请求链路耗时时间,比如:「发送短信、邮件、异步更新等」,这些都是典型的可以通过异步实现的场景。
❞
二、异步的八种实现方式
-
线程Thread
-
Future
-
异步框架CompletableFuture
-
Spring注解@Async
-
Spring ApplicationEvent事件
-
消息队列
-
第三方异步框架,比如Hutool的ThreadUtil
-
Guava异步
三、什么是异步?
首先我们先看一个常见的用户下单的场景:
什么是异步
在同步操作中,我们执行到 「发送短信」 的时候,我们必须等待这个方法彻底执行完才能执行 「赠送积分」 这个操作,如果 「赠送积分」 这个动作执行时间较长,发送短信需要等待,这就是典型的同步场景。
实际上,发送短信和赠送积分没有任何的依赖关系,通过异步,我们可以实现赠送积分
和发送短信
这两个操作能够同时进行,比如:
异步
这就是所谓的异步,是不是非常简单,下面就说说异步的几种实现方式吧。
四、异步编程
4.1 线程异步
public class AsyncThread extends Thread {
@Override
public void run() {
System.out.println("Current thread name:" + Thread.currentThread().getName() + " Send email success!");
}
public static void main(String[] args) {
AsyncThread asyncThread = new AsyncThread();
asyncThread.run();
}
}
当然如果每次都创建一个Thread
线程,频繁的创建、销毁,浪费系统资源,我们可以采用线程池:
private ExecutorService executorService = Executors.newCachedThreadPool();
public void fun() {
executorService.submit(new Runnable() {
@Override
public void run() {
log.info("执行业务逻辑...");
}
});
}
可以将业务逻辑封装到Runnable
或Callable
中,交由线程池来执行。
4.2 Future异步
@Slf4j
public class FutureManager {
public String execute() throws Exception {
ExecutorService executor = Executors.newFixedThreadPool(1);
Future<String> future = executor.submit(new Callable<String>() {
@Override
public String call() throws Exception {
System.out.println(" --- task start --- ");
Thread.sleep(3000);
System.out.println(" --- task finish ---");
return "this is future execute final result!!!