溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用

發(fā)布時間:2023-03-17 11:31:13 來源:億速云 閱讀:126 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要講解了“SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用”吧!

前言

  • 同步編程:在同步編程中,任務(wù)一次執(zhí)行一個,只有當(dāng)一個任務(wù)完成時,下一個任務(wù)才會被解除阻塞。

  • 異步編程:在異步編程中,可以同時執(zhí)行多個任務(wù)。您可以在上一個任務(wù)完成之前轉(zhuǎn)到另一個任務(wù)。

SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用

Spring Boot中,我們可以使用@Async注解來實現(xiàn)異步行為。

實現(xiàn)步驟

1.定義一個異步服務(wù)接口AsyncService.java

public interface AsyncService {

    void asyncMethod() throws InterruptedException;

    Future<String> futureMethod() throws InterruptedException;
}

2.實現(xiàn)定義的接口AsyncServiceImpl.java

@Service
@Slf4j
public class AsyncServiceImpl implements AsyncService  {

    @Async
    @Override
    public void asyncMethod() throws InterruptedException {
        Thread.sleep(3000);
        log.info("Thread: [{}], Calling other service..", Thread.currentThread().getName());
    }

    @Async
    @Override
    public Future<String> futureMethod() throws InterruptedException {
        Thread.sleep(5000);
        log.info("Thread: [{}], Calling other service..", Thread.currentThread().getName());
        return new AsyncResult<>("task Done");
    }
}
  • AsyncServiceImpl 是一個 spring 管理的 bean

  • 您的異步方法必須是公共的,而且是被@Async注解修飾。

  • 返回類型被限制為 void 或 Future

3.定義一個控制器AsyncController.java

@EnableAsync
@RestController
@Slf4j
public class AsyncController {
    @Autowired
    AsyncService asyncService;

    @GetMapping("/async")
    public String asyncCallerMethod() throws InterruptedException {
        long start = System.currentTimeMillis();
        log.info("call async method, thread name: [{}]", Thread.currentThread().getName());
        asyncService.asyncMethod();
        String response = "task completes in :" +
                (System.currentTimeMillis() - start) + "milliseconds";
        return response;
    }

    @GetMapping("/asyncFuture")
    public String asyncFuture() throws InterruptedException, ExecutionException {
        long start = System.currentTimeMillis();
        log.info("call async method, thread name: [{}]", Thread.currentThread().getName());
        Future<String> future = asyncService.futureMethod();
        // 阻塞獲取結(jié)果
        String taskResult = future.get();
        String response = taskResult + "task completes in :" +
                (System.currentTimeMillis() - start) + "milliseconds";
        return response;
    }
}
  • 關(guān)鍵點,需要添加啟用異步的注解@EnableAsync ,當(dāng)然這個注解加在其他地方也ok得。

  • 當(dāng)外部調(diào)用該接口時,asyncMethod()將由默認(rèn)任務(wù)執(zhí)行程序創(chuàng)建的另一個線程執(zhí)行,主線程不需要等待完成異步方法執(zhí)行。

4.運行一下

現(xiàn)在我們運行一下看看,是不是異步返回的。

SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用

SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用

可以看到調(diào)用/async接口,最終一步調(diào)用了方法。

SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用

SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用

調(diào)用/asyncFuture,發(fā)現(xiàn)返回5秒多,難道不是異步的嗎?其實也是異步的,看日志可以看出來,只不過我們返回的是Future,調(diào)用Futrue.get()是阻塞的。

自定義異步任務(wù)執(zhí)行器和異常處理

我們現(xiàn)在看看如果異常方法中報錯了會怎么樣?修改異步代碼如下所示,會拋運行時異常:

SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用

再次執(zhí)行異步接口,如下所示,會使用默認(rèn)的線程池和異常處理。

SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用

我們也可以自定義異步方法的處理異常和異步任務(wù)執(zhí)行器,我們需要配置 AsyncUncaughtExceptionHandler,如下代碼所示:

@Configuration
public class AsynConfiguration extends AsyncConfigurerSupport {
   @Override
   public Executor getAsyncExecutor() {
      ThreadPoolTaskExecutor executor = new 
                ThreadPoolTaskExecutor();
      executor.setCorePoolSize(3);
      executor.setMaxPoolSize(4);
      executor.setThreadNamePrefix("asyn-task-thread-");
      executor.setWaitForTasksToCompleteOnShutdown(true);
      executor.initialize();
      return executor;
  }
  @Override
  public AsyncUncaughtExceptionHandler  
         getAsyncUncaughtExceptionHandler() {
     return new AsyncUncaughtExceptionHandler() {
   
        @Override
        public void handleUncaughtException(Throwable ex, 
           Method method, Object... params) {
           System.out.println("Exception: " + ex.getMessage());
           System.out.println("Method Name: " + method.getName());
           ex.printStackTrace();
        }
    };
  }
}

再次運行,得到的結(jié)果如下:

SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用

@Async如何工作的

必須通過使用 @EnableAsync注解注解主應(yīng)用程序類或任何直接或間接異步方法調(diào)用程序類來啟用異步支持。主要通過代理模式實現(xiàn),默認(rèn)模式是 Proxy,另一種是 AspectJ。代理模式只允許通過代理攔截調(diào)用。永遠(yuǎn)不要從定義它的同一個類調(diào)用異步方法,它不會起作用。

當(dāng)使用 @Async對方法進行注解時,它會根據(jù)“proxyTargetClass”屬性為該對象創(chuàng)建一個代理。當(dāng) spring 執(zhí)行這個方法時,默認(rèn)情況下它會搜索關(guān)聯(lián)的線程池定義。上下文中唯一的 spring 框架 TaskExecutor bean 或名為“taskExecutor”的 Executor bean。如果這兩者都不可解析,默認(rèn)會使用spring框架SimpleAsyncTaskExecutor來處理異步方法的執(zhí)行。

感謝各位的閱讀,以上就是“SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對SpringBoot怎么優(yōu)雅地實現(xiàn)異步調(diào)用這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關(guān)知識點的文章,歡迎關(guān)注!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI