1、線程Thread
2、Future
3、異步框架CompletableFuture
4、Spring注解@Async
5、Spring ApplicationEvent事件
6、消息隊列
7、第三方異步框架,比如Hutool的ThreadUtil
8、Guava異步
首先我們先看一個常見的用戶下單的場景:
在同步操作中,我們執行到 發送短信 的時候,我們必須等待這個方法徹底執行完才能執行 贈送積分 這個操作,如果 贈送積分 這個動作執行時間較長,發送短信需要等待,這就是典型的同步場景。
實際上,發送短信和贈送積分沒有任何的依賴關系,通過異步,我們可以實現贈送積分和發送短信這兩個操作能夠同時進行,比如:
異步
這就是所謂的異步,是不是非常簡單,下面就說說異步的幾種實現方式吧。
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中,交由線程池來執行。
@Slf4jpublic 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!!!
Future的不足之處的包括以下幾點:
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來處理異步任務,如果在某些業務場景我們想自定義自己的異步線程池也是可以的。
/** * 線程池參數配置,多個線程池實現線程池隔離,@Async注解,默認使用系統自定義線程池,可在項目中設置多個線程池,在異步調用的時候,指明需要調用的線程池名稱,比如:@Async("taskName")@EnableAsync@Configurationpublic 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; }}
public interface AsyncService { MessageResult sendSms(String callPrefix, String mobile, String actionType, String content); MessageResult sendEmail(String email, String subject, String content);}@Slf4j@Servicepublic 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直接實現異步。
public class AsyncSendEmailEvent extends ApplicationEvent { /** * 郵箱 **/ private String email; /** * 主題 **/ private String subject; /** * 內容 **/ private String content; /** * 接收者 **/ private String targetUserId;}
@Slf4j@Componentpublic 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重試來幫助我們避免這種異常造成數據不一致問題。
@Slf4j@Componentpublic 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; } }); }}
@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); } }}
@Slf4jpublic 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!"); }}
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(); }});
那么,以上就是本期介紹的分布式服務下實現異步的八種方式了,供您參考。
本文鏈接:http://www.www897cc.com/showinfo-26-16837-0.html聊聊分布式服務下的八種異步實現方式
聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。郵件:2376512515@qq.com
上一篇: 一文搞懂為什么選擇 Java 虛擬線程?
下一篇: Go異步任務解決方案:Asynq