国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

同步 vs 異步性能差十倍!SpringBoot 高吞吐接口實現(xiàn)終極方案

開發(fā)
服務(wù)端的異步或同步對于客戶端而言是不可見的。不會因為服務(wù)端使用了異步,接口的結(jié)果就和同步不一樣了。

Servlet 3.0之前:每一次Http請求都由一個線程從頭到尾處理。

Servlet 3.0之后,提供了異步處理請求:可以先釋放容器分配給請求的線程與相關(guān)資源,減輕系統(tǒng)負擔,從而增加服務(wù)的吞吐量。

在springboot應(yīng)用中,可以有4種方式實現(xiàn)異步接口:

  • AsyncContext
  • Callable
  • WebAsyncTask
  • DeferredResult

第一中AsyncContext是Servlet層級的,比較原生的方式,本文不對此介紹(一般都不使用它,太麻煩了)。本文著重介紹后面三種方式。

特別說明:服務(wù)端的異步或同步對于客戶端而言是不可見的。不會因為服務(wù)端使用了異步,接口的結(jié)果就和同步不一樣了。另外,對于單個請求而言,使用異步接口會導致響應(yīng)時間比同步大,但不特別明顯。具體后文分析。

基于Callable實現(xiàn)

Controller中,返回一個java.util.concurrent.Callable包裝的任何值,都表示該接口是一個異步接口:

@GetMapping("/testCallAble")
public Callable<String> testCallAble() {
    return () -> {
        Thread.sleep(40000);
        return "hello";
    };
}

服務(wù)器端的異步處理對客戶端來說是不可見的。例如,上述接口,最終返回的客戶端的是一個String,和同步接口中,直接返回String的效果是一樣的。

Callable 處理過程如下:

  • 控制器返回一個 Callable。
  • Spring MVC 調(diào)用 request.startAsync() 并將 Callable 提交給 AsyncTaskExecutor 以在單獨的線程中進行處理。
  • 同時, DispatcherServlet 和所有過濾器退出 Servlet 容器線程,但response保持打開狀態(tài)。
  • 最終 Callable 產(chǎn)生結(jié)果,Spring MVC將請求分派回Servlet容器以完成處理。
  • 再次調(diào)用 DispatcherServlet,并使用 Callable 異步生成的返回值繼續(xù)處理。

Callable默認使用SimpleAsyncTaskExecutor類來執(zhí)行,這個類非常簡單而且沒有重用線程。在實踐中,需要使用AsyncTaskExecutor類來對線程進行配置。

基于WebAsyncTask實現(xiàn)

Spring提供的WebAsyncTask是對Callable的包裝,提供了更強大的功能,比如:處理超時回調(diào)、錯誤回調(diào)、完成回調(diào)等。本質(zhì)上,和Callable區(qū)別不大,但是由于它額外封裝了一些事件的回調(diào),所有,通常都使用WebAsyncTask而不是Callable:

@GetMapping("/webAsyncTask")
public WebAsyncTask<String> webAsyncTask() {
    WebAsyncTask<String> result = new WebAsyncTask<>(30003, () -> {
        return "success";
    });
    result.onTimeout(() -> {
        log.info("timeout callback");
        return "timeout callback";
    });
    result.onCompletion(() -> log.info("finish callback"));
    return result;
}

這里額外提一下,WebAsyncTask可以配置一個超時時間,這里配置的超時時間比全局配置的超時時間優(yōu)先級都高(會覆蓋全局配置的超時時間)。

基于DeferredResult實現(xiàn)

DeferredResult使用方式與Callable類似,但在返回結(jié)果時不一樣,它返回的時實際結(jié)果可能沒有生成,實際的結(jié)果可能會在另外的線程里面設(shè)置到DeferredResult中去。

//定義一個全局的變量,用來存儲DeferredResult對象
private Map<String, DeferredResult<String>> deferredResultMap = new ConcurrentHashMap<>();

@GetMapping("/testDeferredResult")
public DeferredResult<String> testDeferredResult(){
    DeferredResult<String> deferredResult = new DeferredResult<>();
    deferredResultMap.put("test", deferredResult);
    return deferredResult;
}

如果調(diào)用以上接口,會發(fā)現(xiàn)客戶端的請求一直是在pending狀態(tài)——等待后端響應(yīng)。這里,我簡單的將該接口返回的DeferredResult對象存放在了一個Map集合中,實際應(yīng)用中可以設(shè)計一個對象管理器來統(tǒng)一管理這些個對象。

注意:要考慮定時輪詢(或其他方式)這些對象,將已經(jīng)處理過或無效的DeferredResult對象清理掉(DeferredResult.isSetOrExpired方法可以判斷是否還有效),避免內(nèi)存泄露。

這里我又寫了一個接口,模擬:

@GetMapping("/testSetDeferredResult")
public String testSetDeferredResult() throws InterruptedException {
    DeferredResult<String> deferredResult = deferredResultMap.get("test");
    boolean flag = deferredResult.setResult("testSetDeferredResult");
    if(!flag){
        log.info("結(jié)果已經(jīng)被處理,此次操作無效");
    }
    return "ok";
}

其他線程修改DeferredResult的值:首先是從之前存放DeferredResult的map中拿到DeferredResult的值,然后設(shè)置它的返回值。當執(zhí)行deferredResult.setResult之后,可以看到之前pending狀態(tài)的接口完成了響應(yīng),得到的結(jié)果,就是這里設(shè)置的值。

這里也額外說下:在返回DeferredResult時也可以設(shè)置超時時間,這個時間的優(yōu)先級也是大于全局設(shè)置的。另外,判斷DeferredResult是否有效,只是一個簡單的判斷,實際中判斷有效的并不一定是有效的(比如:客戶端取消了請求,服務(wù)端是不知道的),但是一般判斷為無效的,那肯定是無效了。

DeferredResult 處理過程如下:

  • 控制器返回一個 DeferredResult 并將其保存在可以訪問的內(nèi)存隊列或列表中。
  • Spring MVC 調(diào)用 request.startAsync() 。
  • 同時,DispatcherServlet 和所有配置的過濾器退出請求處理線程,但響應(yīng)保持打開狀態(tài)。
  • 應(yīng)用程序從某個線程設(shè)置 DeferredResult,Spring MVC 將請求分派回 Servlet 容器。
  • 再次調(diào)用 DispatcherServlet,并使用異步生成的返回值繼續(xù)處理。

提供一個線程池

異步請求,不會一直占用請求的主線程(tomcat容器中處理請求的線程),而是通過一個其他的線程來處理異步任務(wù)。也正是如此,在相同的最大請求數(shù)配置下,異步請求由于迅速的釋放了主線程,所以才能提高吞吐量。

這里提到一個其他線程,那么這個其他線程我們一般都不適用默認的,都是根據(jù)自身情況提供一個線程池供異步請求使用:(我給的參數(shù)都是測試用的,實際中不可照搬)

@Bean("mvcAsyncTaskExecutor")
public AsyncTaskExecutor asyncTaskExecutor() {
    ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
    // 線程池維護線程的最少數(shù)量
    // asyncServiceExecutor.setCorePoolSize(Runtime.getRuntime().availableProcessors() + 1);
    executor.setCorePoolSize(5);
    // 線程池維護線程的最大數(shù)量
    executor.setMaxPoolSize(10);
    // 線程池所使用的緩沖隊列
    executor.setQueueCapacity(10);
    //   asyncServiceExecutor.prefersShortLivedTasks();
    executor.setThreadNamePrefix("fyk-mvcAsyncTask-Thread-");
//      asyncServiceExecutor.setBeanName("TaskId" + taskId);
    //  asyncServiceExecutor.setKeepAliveSeconds(20);
    //調(diào)用者執(zhí)行
    //   asyncServiceExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
    executor.setRejectedExecutionHandler(new ThreadPoolExecutor.DiscardPolicy());
    // 線程全部結(jié)束才關(guān)閉線程池
    executor.setWaitForTasksToCompleteOnShutdown(true);
    // 如果超過60s還沒有銷毀就強制銷毀,以確保應(yīng)用最后能夠被關(guān)閉,而不是阻塞住
    executor.setAwaitTerminationSeconds(30);
    executor.initialize();

    return executor;
}

把這個線程池配置設(shè)置到異步請求配置中:

@Configuration
public class FykWebMvcConfigurer implements WebMvcConfigurer {

    @Autowired
    @Qualifier("mvcAsyncTaskExecutor")
    private AsyncTaskExecutor asyncTaskExecutor;

    @Override
    public void configureAsyncSupport(AsyncSupportConfigurer configurer) {
        //異步操作的超時時間,值為0或者更小,表示永不超時
        configurer.setDefaultTimeout(60001);
        configurer.setTaskExecutor(asyncTaskExecutor);
    }
}

什么時候使用異步請求

異步請求能提高吞吐量,這個是建立在相同配置(這里的配置指的是:最大連接數(shù)、最大工作線程數(shù))的情況下。因此并不是說任何接口都可以使用異步請求。

比如:一個請求是進行大量的計算(總之就是在處理這個請求的業(yè)務(wù)方法時CPU是沒有休息的),這種情況使用異步請求就沒有多大意義了,因為這時的異步請求只是把一個任務(wù)從tomcat的工作線程搬到了另一個線程罷了。直接調(diào)大最大工作線程數(shù)配置也能到達要求。

所以,真正使用異步請求的場景應(yīng)該是該請求的業(yè)務(wù)代碼中,大量的時間CPU是休息的(比如:在業(yè)務(wù)代碼中請求其他系統(tǒng)的接口,在其他系統(tǒng)響應(yīng)之前,CPU是阻塞等待的),這個時候使用異步請求,就可以釋放tomcat的工作線程,讓釋放的工作線程可以處理其他的請求,從而提高吞吐量。

由于異步請求增加了更多的線程切換(同步請求是同一個工作線程一直處理),所以理論上會增加接口的耗時。但,這個耗時很短很短。

責任編輯:趙寧寧 來源: 程序員小富
相關(guān)推薦

2025-07-18 09:40:44

2024-09-12 15:24:29

2024-09-09 14:12:38

2025-03-13 11:59:00

2025-06-05 04:22:00

SQL性能索引

2023-09-07 11:29:36

API開發(fā)

2025-05-26 00:02:00

TypeScriptGo 語言前端

2024-10-09 11:31:51

2025-07-08 09:33:08

2009-12-15 21:49:05

2025-10-10 05:56:11

2022-04-28 07:31:41

Springkafka數(shù)據(jù)量

2022-09-15 16:59:46

人工智能空調(diào)能源

2017-09-26 14:56:57

MongoDBLBS服務(wù)性能

2016-07-07 15:38:07

京東

2021-04-13 14:25:41

架構(gòu)運維技術(shù)

2009-11-19 08:46:16

Windows 7系統(tǒng)驅(qū)動

2025-06-05 03:00:00

Spring異步接口

2025-08-12 07:46:29

2021-09-13 10:25:35

開發(fā)技能代碼
點贊
收藏

51CTO技術(shù)棧公眾號

a黄色片在线观看| 精品69视频一区二区三区Q| 在线观看视频91| 亚洲精品少妇久久久久久| 欧美高清在线视频| 美女黄色片视频| 成人免费视频视频在线观看免费| 可以免费看的黄色网址| 蜜臀精品久久久久久蜜臀| 日本午夜精品电影| 免费高清视频精品| 2021狠狠干| 成人av网址在线| 成人中文字幕av| 国产精品国产三级国产有无不卡 | 日韩极品视频在线观看| 麻豆极品一区二区三区| 午夜精品一区二区三区在线观看| 视频一区欧美日韩| 亚洲高清乱码| 懂色av一区二区三区免费观看 | 欧美激情一区二区三区久久久| 狠狠色综合色区| 亚洲h在线观看| gogo高清午夜人体在线| www日韩av| 欧美极品免费| 久久精品人人爽| 三年中国国语在线播放免费| 91精品综合视频| 国产黄色成人av| 中文字幕av在线播放| 91成人福利在线| 亚洲伊人精品酒店| 高清欧美性猛交xxxx黑人猛交| 精品精品国产毛片在线看| 日本一区二区在线播放| 欧美1区3d| 黄色录像特级片| |精品福利一区二区三区| 国产美女性感在线观看懂色av| 亚洲精品97久久| 亚洲日本视频在线| 91观看网站| 久久国产精品免费| www.xxx黄| 91精品国产麻豆| 国产亚洲久久| 18成人在线| 国产不卡一区视频| 在线免费国产| 亚洲情综合五月天| 欧美精品一二| 亚洲电影一二三区| 中文字幕不卡一区| 国产鲁鲁视频在线观看特色| 久久国产加勒比精品无码| 色天天久久综合婷婷女18| 亚洲国产午夜伦理片大全在线观看网站 | 538在线视频| 欧美精品在线免费| 欧美日韩ab| 成人免费网站入口| 偷拍一区二区三区四区| 最近高清中文在线字幕在线观看1| 国a精品视频大全| 亚洲免费在线| 国产免费视频| 欧美成人欧美edvon| 永久免费精品视频| 日本黄网免费一区二区精品| 中文字幕一区不卡| 亚洲精品**中文毛片| 91久久夜色精品国产网站| 高清不卡一区二区在线| 国产在线视频网| 久久久免费av| 国产一区激情在线| 97电影在线观看| 久久久久久久91| 黄网站免费久久| a天堂中文在线88| 欧美综合在线观看| 成人高清免费观看| 曰本三级在线| 成人在线激情视频| 91麻豆免费观看| 日本高清在线观看视频| 国产精品久久久久久久久男| aa级大片欧美| 黄色美女视频在线观看| 成人性生交大片免费观看嘿嘿视频| 97久久精品人人做人人爽50路| 99视频免费在线观看| 国产精品综合网站| 国产色综合一区| av有声小说一区二区三区| 久久一区二区三区av| 欧美性猛交xxxxx水多| 亚洲综合小说图片| 久久精品一区二| 亚洲精品视频中文字幕| 亚洲深夜影院| 成年在线观看免费人视频| 日韩av免费看网站| 国产精品久久网站| 国产一区影院| 少妇久久久久久被弄到高潮| 91精品欧美一区二区三区综合在| 婷婷久久一区| gogogo影视剧免费观看在线观看| 欧美日韩xxx| 26uuuu精品一区二区| 成人免费在线观看视频| 欧美日韩亚洲国产成人| 日韩久久久久久| 麻豆九一精品爱看视频在线观看免费| 免费人成在线观看网站| 成人欧美在线观看| 午夜精品久久久久久久久久 | 成人福利网址| 美女福利精品视频| www.视频一区| 99蜜月精品久久91| 亚洲美免无码中文字幕在线| 日韩av网址在线| 久久99在线观看| 亚洲性受xxx喷奶水| 国产成人生活片| 中文字幕在线看视频国产欧美| 国产不卡免费视频| 成人黄色免费观看| 欧美 国产 小说 另类| 欧美巨大黑人极品精男| 国产精品天天摸av网| 懂色av一区二区| 国产小黄视频| 国产精品一久久香蕉国产线看观看| 亚洲二区在线观看| 久久久久久久久久久久久久| 国产区视频在线| 日韩精品不卡| 国产亚洲在线播放| 国产午夜精品久久久久久久| 精品精品精品| 在线观看视频污| 久99久视频| 亚洲情综合五月天| 久久综合九色综合97婷婷女人| 国产精品白丝av嫩草影院| 国产一级性片| 99热国产免费| 精品成人免费观看| 成人成人成人在线视频| 北条麻妃一区二区三区在线| 米奇.777.com| 国产成人一区二区三区免费看| 日韩一区二区不卡| av中文字幕在线不卡| 亚洲尤物av| 麻豆传媒在线完整视频| av一区二区三区免费观看| 欧美放荡办公室videos4k| 一区二区三区在线视频观看| 亚洲欧洲日韩| 亚洲黄色中文字幕| 日本在线免费观看视频| 久久66热这里只有精品| 亚洲片国产一区一级在线观看| 中文无字幕一区二区三区| 一区二区在线| 免费成人美女女| 欧美写真视频一区| 色狠狠久久av五月综合|| 欧美成人一二三| 欧美性猛交xxxx| 国产一区二区精品久久91| 999在线精品| 毛片在线看片| 国产成人无码一二三区视频| 96sao精品视频在线观看| 亚洲精品中文字| 亚洲一区二区三区激情| 麻豆91精品视频| 不卡在线一区二区| 三级成人黄色影院| 超碰在线人人| 男人的天堂avav| 亚洲va国产va天堂va久久| 色阁综合伊人av| 欧美日韩成人一区二区| 国产日韩亚洲欧美综合| 久久综合亚州| 欧美美乳视频| 巨茎人妖videos另类| 香蕉97视频观看在线观看| 777久久久精品一区二区三区| 国产一区在线免费| 奇米一区二区三区四区久久| 日韩精品免费在线视频|