8 種異步實現方式歸納總結

前言:異步執行對於開發者來說並不陌生,在實際的開發過程中,很多場景多會使用到異步,相比同步執行,異步可以大大縮短請求鏈路耗時時間,比如:「發送短信、郵件、異步更新等」,這些都是典型的可以通過異步實現的場景。

一、異步的八種實現方式

1、線程 Thread

2、Future

3、異步框架 CompletableFuture

4、Spring 註解 @Async

5、Spring ApplicationEvent 事件

6、消息隊列

7、第三方異步框架,比如 Hutool 的 ThreadUtil

8、Guava 異步

二、什麼是異步?

首先我們先看一個常見的用戶下單的場景:

什麼是異步?

在同步操作中,我們執行到 發送短信 的時候,我們必須等待這個方法徹底執行完才能執行 贈送積分 這個操作,如果 贈送積分 這個動作執行時間較長,發送短信需要等待,這就是典型的同步場景。

實際上,發送短信和贈送積分沒有任何的依賴關係,通過異步,我們可以實現贈送積分和發送短信這兩個操作能夠同時進行,比如:

異步

這就是所謂的異步,是不是非常簡單,下面就說說異步的幾種實現方式吧。

三、異步編程

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 中,交由線程池來執行。

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!!!";
            }
        });
        //這裏需要返回值時會阻塞主線程
        String result = future.get();
        log.info("Future get result: {}", result);
        return result;
    }
    @SneakyThrows
    public static void main(String[] args) {
        FutureManager manager = new FutureManager();
        manager.execute();
    }
}

輸出結果:

--- task start --- 
 --- task finish ---
 Future get result: this is future execute final result!!!

(1) Future 的不足之處

Future 的不足之處的包括以下幾點:

**無法被動接收異步任務的計算結果:**雖然我們可以主動將異步任務提交給線程池中的線程來執行,但是待異步任務執行結束之後,主線程無法得到任務完成與否的通知,它需要通過 get 方法主動獲取任務執行的結果。

**Future 件彼此孤立:**有時某一個耗時很長的異步任務執行結束之後,你想利用它返回的結果再做進一步的運算,該運算也會是一個異步任務,兩者之間的關係需要程序開發人員手動進行綁定賦予,Future 並不能將其形成一個任務流(pipeline),每一個 Future 都是彼此之間都是孤立的,所以纔有了後面的 CompletableFuture,CompletableFuture 就可以將多個 Future 串聯起來形成任務流。

**Futrue 沒有很好的錯誤處理機制:**截止目前,如果某個異步任務在執行發的過程中發生了異常,調用者無法被動感知,必須通過捕獲 get 方法的異常才知曉異步任務執行是否出現了錯誤,從而在做進一步的判斷處理。

3、CompletableFuture 實現異步

public class CompletableFutureCompose {
    /**
     * thenAccept子任務和父任務公用同一個線程
     */
    @SneakyThrows
    public static void thenRunAsync() {
        CompletableFuture<Integer> cf1 = CompletableFuture.supplyAsync(() -> {
            System.out.println(Thread.currentThread() + " cf1 do something....");
            return 1;
        });
        CompletableFuture<Void> cf2 = cf1.thenRunAsync(() -> {
            System.out.println(Thread.currentThread() + " cf2 do something...");
        });
        //等待任務1執行完成
        System.out.println("cf1結果->" + cf1.get());
        //等待任務2執行完成
        System.out.println("cf2結果->" + cf2.get());
    }
    public static void main(String[] args) {
        thenRunAsync();
    }
}

我們不需要顯式使用 ExecutorService,CompletableFuture 內部使用了 ForkJoinPool 來處理異步任務,如果在某些業務場景我們想自定義自己的異步線程池也是可以的。

4、Spring 的 @Async 異步

(1)自定義異步線程池

/**
 * 線程池參數配置,多個線程池實現線程池隔離,@Async註解,默認使用系統自定義線程池,可在項目中設置多個線程池,在異步調用的時候,指明需要調用的線程池名稱,比如:@Async("taskName")
@EnableAsync
@Configuration
public class TaskPoolConfig {
    /**
     * 自定義線程池
     *
     **/
    @Bean("taskExecutor")
    public Executor taskExecutor() {
        //返回可用處理器的Java虛擬機的數量 12
        int i = Runtime.getRuntime().availableProcessors();
        System.out.println("系統最大線程數  :" + i);
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        //核心線程池大小
        executor.setCorePoolSize(16);
        //最大線程數
        executor.setMaxPoolSize(20);
        //配置隊列容量,默認值爲Integer.MAX_VALUE
        executor.setQueueCapacity(99999);
        //活躍時間
        executor.setKeepAliveSeconds(60);
        //線程名字前綴
        executor.setThreadNamePrefix("asyncServiceExecutor -");
        //設置此執行程序應該在關閉時阻止的最大秒數,以便在容器的其餘部分繼續關閉之前等待剩餘的任務完成他們的執行
        executor.setAwaitTerminationSeconds(60);
        //等待所有的任務結束後再關閉線程池
        executor.setWaitForTasksToCompleteOnShutdown(true);
        return executor;
    }
}

(2) AsyncService

public interface AsyncService {
    MessageResult sendSms(String callPrefix, String mobile, String actionType, String content);
    MessageResult sendEmail(String email, String subject, String content);
}
@Slf4j
@Service
public class AsyncServiceImpl implements AsyncService {
    @Autowired
    private IMessageHandler mesageHandler;
    @Override
    @Async("taskExecutor")
    public MessageResult sendSms(String callPrefix, String mobile, String actionType, String content) {
        try {
            Thread.sleep(1000);
            mesageHandler.sendSms(callPrefix, mobile, actionType, content);
        } catch (Exception e) {
            log.error("發送短信異常 -> ", e)
        }
    }
    @Override
    @Async("taskExecutor")
    public sendEmail(String email, String subject, String content) {
        try {
            Thread.sleep(1000);
            mesageHandler.sendsendEmail(email, subject, content);
        } catch (Exception e) {
            log.error("發送email異常 -> ", e)
        }
    }
}

在實際項目中, 使用 @Async 調用線程池,推薦等方式是是使用自定義線程池的模式,不推薦直接使用 @Async 直接實現異步。

5、Spring ApplicationEvent 事件實現異步

(1)定義事件

public class AsyncSendEmailEvent extends ApplicationEvent {
    /**
     * 郵箱
     **/
    private String email;
   /**
     * 主題
     **/
    private String subject;
    /**
     * 內容
     **/
    private String content;
    /**
     * 接收者
     **/
    private String targetUserId;
}

(2)定義事件處理器

@Slf4j
@Component
public class AsyncSendEmailEventHandler implements ApplicationListener<AsyncSendEmailEvent> {
    @Autowired
    private IMessageHandler mesageHandler;
    @Async("taskExecutor")
    @Override
    public void onApplicationEvent(AsyncSendEmailEvent event) {
        if (event == null) {
            return;
        }
        String email = event.getEmail();
        String subject = event.getSubject();
        String content = event.getContent();
        String targetUserId = event.getTargetUserId();
        mesageHandler.sendsendEmailSms(email, subject, content, targerUserId);
      }
}

另外,可能有些時候採用 ApplicationEvent 實現異步的使用,當程序出現異常錯誤的時候,需要考慮補償機制,那麼這時候可以結合 Spring Retry 重試來幫助我們避免這種異常造成數據不一致問題。

6、消息隊列

(1)回調事件消息生產者

@Slf4j
@Component
public class CallbackProducer {
    @Autowired
    AmqpTemplate amqpTemplate;
    public void sendCallbackMessage(CallbackDTO allbackDTO, final long delayTimes) {
        log.info("生產者發送消息,callbackDTO,{}", callbackDTO);
        amqpTemplate.convertAndSend(CallbackQueueEnum.QUEUE_GENSEE_CALLBACK.getExchange(), CallbackQueueEnum.QUEUE_GENSEE_CALLBACK.getRoutingKey(), JsonMapper.getInstance().toJson(genseeCallbackDTO), new MessagePostProcessor() {
            @Override
            public Message postProcessMessage(Message message) throws AmqpException {
                //給消息設置延遲毫秒值,通過給消息設置x-delay頭來設置消息從交換機發送到隊列的延遲時間
                message.getMessageProperties().setHeader("x-delay", delayTimes);
                message.getMessageProperties().setCorrelationId(callbackDTO.getSdkId());
                return message;
            }
        });
    }
}

(2)回調事件消息消費者

@Slf4j
@Component
@RabbitListener(queues = "message.callback", containerFactory = "rabbitListenerContainerFactory")
public class CallbackConsumer {
    @Autowired
    private IGlobalUserService globalUserService;
    @RabbitHandler
    public void handle(String json, Channel channel, @Headers Map<String, Object> map) throws Exception {
        if (map.get("error") != null) {
            //否認消息
            channel.basicNack((Long) map.get(AmqpHeaders.DELIVERY_TAG), false, true);
            return;
        }
        try {
            CallbackDTO callbackDTO = JsonMapper.getInstance().fromJson(json, CallbackDTO.class);
            //執行業務邏輯
            globalUserService.execute(callbackDTO);
            //消息消息成功手動確認,對應消息確認模式acknowledge-mode: manual
            channel.basicAck((Long) map.get(AmqpHeaders.DELIVERY_TAG), false);
        } catch (Exception e) {
            log.error("回調失敗 -> {}", e);
        }
    }
}

7、ThreadUtil 異步工具類

@Slf4j
public class ThreadUtils {
    public static void main(String[] args) {
        for (int i = 0; i < 3; i++) {
            ThreadUtil.execAsync(() -> {
                ThreadLocalRandom threadLocalRandom = ThreadLocalRandom.current();
                int number = threadLocalRandom.nextInt(20) + 1;
                System.out.println(number);
            });
            log.info("當前第:" + i + "個線程");
        }
        log.info("task finish!");
    }
}

8、Guava 異步

Guava 的 ListenableFuture 顧名思義就是可以監聽的 Future,是對 java 原生 Future 的擴展增強。我們知道 Future 表示一個異步計算任務,當任務完成時可以得到計算結果。如果我們希望一旦計算完成就拿到結果展示給用戶或者做另外的計算,就必須使用另一個線程不斷的查詢計算狀態。這樣做,代碼複雜,而且效率低下。使用「Guava ListenableFuture」可以幫我們檢測 Future 是否完成了,不需要再通過 get() 方法苦苦等待異步的計算結果,如果完成就自動調用回調函數,這樣可以減少併發程序的複雜度。

ListenableFuture 是一個接口,它從 jdk 的 Future 接口繼承,添加了 void addListener(Runnable listener, Executor executor) 方法。

我們看下如何使用 ListenableFuture。首先需要定義 ListenableFuture 的實例:

 ListeningExecutorService executorService = MoreExecutors.listeningDecorator(Executors.newCachedThreadPool());
        final ListenableFuture<Integer> listenableFuture = executorService.submit(new Callable<Integer>() {
            @Override
            public Integer call() throws Exception {
                log.info("callable execute...")
                TimeUnit.SECONDS.sleep(1);
                return 1;
            }
        });

首先通過 MoreExecutors 類的靜態方法 listeningDecorator 方法初始化一個 ListeningExecutorService 的方法,然後使用此實例的 submit 方法即可初始化 ListenableFuture 對象。

ListenableFuture 要做的工作,在 Callable 接口的實現類中定義,這裏只是休眠了 1 秒鐘然後返回一個數字 1,有了 ListenableFuture 實例,可以執行此 Future 並執行 Future 完成之後的回調函數。

 Futures.addCallback(listenableFuture, new FutureCallback<Integer>() {
    @Override
    public void onSuccess(Integer result) {
        //成功執行...
        System.out.println("Get listenable future's result with callback " + result);
    }
    @Override
    public void onFailure(Throwable t) {
        //異常情況處理...
        t.printStackTrace();
    }
});

那麼,以上就是本期介紹的實現異步的 8 種方式了。

本文由 Readfog 進行 AMP 轉碼,版權歸原作者所有。
來源https://mp.weixin.qq.com/s/dXqlmv82FDnYKNa9dFCX0g