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

SpringBoot 異步接口實(shí)現(xiàn):提高系統(tǒng)的吞吐量

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

前言

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

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

在springboot應(yīng)用中,可以有4種方式實(shí)現(xiàn)異步接口(至于ResponseBodyEmitter、SseEmitter、StreamingResponseBody,不在本文介紹內(nèi),之后新寫文章介紹):

  • AsyncContext
  • Callable
  • WebAsyncTask
  • DeferredResult

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

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

基于Callable實(shí)現(xiàn)

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

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

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

Callable 處理過程如下:

控制器返回一個(gè) Callable 。

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

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

基于WebAsyncTask實(shí)現(xiàn)

Spring提供的WebAsyncTask是對(duì)Callable的包裝,提供了更強(qiáng)大的功能,比如:處理超時(shí)回調(diào)、錯(cuò)誤回調(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可以配置一個(gè)超時(shí)時(shí)間,這里配置的超時(shí)時(shí)間比全局配置的超時(shí)時(shí)間優(yōu)先級(jí)都高(會(huì)覆蓋全局配置的超時(shí)時(shí)間)。

基于DeferredResult實(shí)現(xiàn)

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

//定義一個(gè)全局的變量,用來存儲(chǔ)DeferredResult對(duì)象
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)用以上接口,會(huì)發(fā)現(xiàn)客戶端的請求一直是在pending狀態(tài)——等待后端響應(yīng)。這里,我簡單的將該接口返回的DeferredResult對(duì)象存放在了一個(gè)Map集合中,實(shí)際應(yīng)用中可以設(shè)計(jì)一個(gè)對(duì)象管理器來統(tǒng)一管理這些個(gè)對(duì)象。

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

這里我又寫了一個(gè)接口,模擬:

@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è)置它的返回值。當(dāng)執(zhí)行deferredResult.setResult之后,可以看到之前pending狀態(tài)的接口完成了響應(yīng),得到的結(jié)果,就是這里設(shè)置的值。

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

DeferredResult 處理過程如下:

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

提供一個(gè)線程池

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

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

@Bean("mvcAsyncTaskExecutor")
public AsyncTaskExecutor asyncTaskExecutor() {
    ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
    // 線程池維護(hù)線程的最少數(shù)量
    // asyncServiceExecutor.setCorePoolSize(Runtime.getRuntime().availableProcessors() + 1);
    executor.setCorePoolSize(5);
    // 線程池維護(hù)線程的最大數(shù)量
    executor.setMaxPoolSize(10);
    // 線程池所使用的緩沖隊(duì)列
    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還沒有銷毀就強(qiáng)制銷毀,以確保應(yīng)用最后能夠被關(guān)閉,而不是阻塞住
    executor.setAwaitTerminationSeconds(30);
    executor.initialize();

    return executor;
}

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

@Configuration
public class FykWebMvcConfigurer implements WebMvcConfigurer {

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

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

什么時(shí)候使用異步請求

異步請求能提高吞吐量,這個(gè)是建立在相同配置(這里的配置指的是:最大連接數(shù)、最大工作線程數(shù))的情況下。因此并不是說任何接口都可以使用異步請求。比如:一個(gè)請求是進(jìn)行大量的計(jì)算(總之就是在處理這個(gè)請求的業(yè)務(wù)方法時(shí)CPU是沒有休息的),這種情況使用異步請求就沒有多大意義了,因?yàn)檫@時(shí)的異步請求只是把一個(gè)任務(wù)從tomcat的工作線程搬到了另一個(gè)線程罷了。

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

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

責(zé)任編輯:趙寧寧 來源: 碼猿技術(shù)專欄
相關(guān)推薦

2024-09-09 14:12:38

2023-02-09 08:57:11

Callable異步java

2025-07-08 09:33:08

2025-07-18 09:40:44

2025-05-23 08:37:26

2025-04-16 08:25:00

2024-06-28 09:39:58

2021-12-26 00:03:27

響應(yīng)式編程異步

2025-03-28 01:03:46

高并發(fā)技術(shù)異步

2023-08-03 14:18:29

Rust阻塞函數(shù)

2023-11-07 15:11:46

Kafka技巧

2022-11-11 10:13:06

數(shù)據(jù)庫內(nèi)存Milvus

2025-06-05 03:00:00

Spring異步接口

2024-09-14 11:31:27

@AsyncSpring異步

2024-05-23 16:41:40

2013-04-19 09:45:20

AMPLabHadoopHDFS

2024-01-19 13:42:00

模型訓(xùn)練

2019-08-20 00:20:47

TCPHOL吞吐量

2013-04-25 10:38:40

思科存儲(chǔ)交換機(jī)

2024-06-06 16:15:00

點(diǎn)贊
收藏

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

国产视频一区二| 国产探花一区二区| 欧美日韩成人综合在线一区二区| 色哟哟在线观看| 91久久国产综合久久| 四虎精品成人影院观看地址| 日韩欧美成人区| 日韩在线无毛| 欧美系列在线观看| 欧美xxxxxx| 日本不卡视频在线播放| 欧美先锋资源| 亚洲一区二区三区免费看| 精品中文字幕一区二区| 在线观看免费黄网站| 欧洲一区在线观看| 暖暖成人免费视频| 国产三级精品网站| 另类中文字幕网| 精品久久一二三| 午夜精品久久一牛影视| 伦xxxx在线| 18性欧美xxxⅹ性满足| 美女毛片在线看| 欧美一区二区视频在线观看| 国产精品实拍| 日本成人黄色片| 久久精品免费一区二区三区| 黄色网络在线观看| 欧美午夜精品久久久| 亚洲瘦老头同性70tv| 日韩精品另类天天更新| 91欧美一区二区| 理论不卡电影大全神| 国产欧美韩国高清| 久久久精品免费观看| 影音先锋在线播放| 日韩精品在线免费观看视频| 看全色黄大色大片免费久久久| 久久综合狠狠综合久久综青草| 99在线精品免费| 成人香蕉视频| 久久久久久久有限公司| 欧美日韩一级在线| 91久久精品午夜一区二区| 国产成人在线中文字幕| 成人免费在线小视频| 亚洲无亚洲人成网站77777| 天堂日韩电影| 香港经典三级在线| 欧美日韩第一页| 激情欧美一区二区三区| 免费av手机在线观看| 国产亚洲精品久久久优势 | 欧美久久久久久久久| 99久久99久久精品国产片桃花| 亚洲国产精品无码av| 久久99国产精品久久| 成人免费高清在线播放| 国产日韩在线观看av| 日韩欧美黄色动漫| 一区二区黄色| 欧美理论片在线播放| 3d动漫啪啪精品一区二区免费 | 亚洲欧美久久久久| 亚洲韩国欧洲国产日产av| 国产最新精品精品你懂的| 欧美色网一区| 日本爱爱免费视频| 91免费国产视频| 狠狠色狠狠色综合日日五| 亚洲精品一二| 欧美影院视频| 人人爽人人av| 日韩中文在线视频| 国产精品色呦呦| 久久性感美女视频| 国产午夜精品一区理论片| 久久久久久网站| 精品久久久久久久中文字幕| 激情欧美一区二区三区在线观看| 国产成年精品| 久久久亚洲精华液精华液精华液 | 欧美1区2区视频| 欧美韩日亚洲| 亚洲一级免费在线观看| 精品一区二区久久久久久久网站| 精品视频偷偷看在线观看| 自拍偷拍国产精品| 久久精品国产亚洲高清剧情介绍 | 成人啪啪免费看| 亚洲精品一区久久久久久| 亚洲综合精品久久| 亚洲欧美清纯在线制服| 日本电影在线观看| 污视频网站在线观看| 中文字幕无码精品亚洲35| 97视频中文字幕| 欧美黄色性视频| 国产精品一二一区| 激情五月色综合国产精品| 免费在线黄色网址| 国产精品久久久久久久久久免费 | www.涩涩涩| 欧美日韩中文字幕在线播放 | 久久精品女人天堂av免费观看| 欧美vide| 黄色一级视频播放| 成人精品一二区| 久久久久久欧美| 欧美二区乱c黑人| 亚洲精品国产拍免费91在线| 欧美日韩中字一区| 一区二区三区在线观看欧美| 26uuu成人网一区二区三区| 国产一区二区按摩在线观看| 免费视频最近日韩| 久久福利精品| 香蕉国产成人午夜av影院| 精品五月天堂| 国产一区二区三区黄网站| 美女一区网站| 精品中文视频| 国产精品一区二区免费福利视频| 有色激情视频免费在线| 最新在线你懂的| 国产69久久| 最猛黑人系列在线播放| 国产资源在线看| 日本中文字幕中出在线| 国产四区在线观看| 国产福利视频一区二区| 亚洲精品久久久久久久久久久久| 亚洲视频专区在线| 久久久久国产一区二区三区| 国产精品亚洲精品| 四虎永久国产精品| 欧美成人福利在线观看| 在线观看国产v片| 末成年女av片一区二区下载| 欧美色图婷婷| 三级欧美韩日大片在线看| 成人aa视频在线观看| 狠狠躁天天躁日日躁欧美| 欧美在线播放高清精品| 亚洲精品乱码久久久久久金桔影视| 欧美精品少妇videofree| 91色在线视频| 国产精品啪啪啪视频| 日韩精品在线观看一区二区| 美日韩在线视频| 91精品久久久久久久久久另类| 日本一区视频在线观看| 国产videos| 国产福利片在线观看| 久久一区二区三区喷水| 久久成人免费网站| 亚洲成av人片一区二区三区 | 精品网站aaa| 国产麻豆91精品| 欧美日韩精品专区| 欧美伦理91i| 欧美 日韩 国产精品| 麻豆av一区二区三区| 日韩 欧美 高清| 一本色道久久加勒比88综合| 校园春色亚洲色图| 亚洲成人原创| 精品久久久视频| 亚洲精品美女在线| 波多野结衣精品久久| 一个人看的www一区| 日韩av资源网| 亚洲女爱视频在线| 亚洲免费中文字幕| 成人欧美在线观看| 国产精品入口免费软件| 韩漫成人漫画| 国产福利91精品一区二区三区| 欧美另类久久久品| 国产精品igao视频| 96久久久久久| 欧美在线免费| 91精品国产一区二区| 国产精品亚洲综合| 国产在线视频网| 韩日成人在线| 欧美私模裸体表演在线观看| 国产美女精品久久久| 欧美大胆的人体xxxx| 日韩电影在线免费| 中文字幕在线观看日韩| 99热手机在线观看| 国产一区99| 欧美中文字幕一二三区视频| 北条麻妃高清一区| 欧美xxxx做受欧美护士| 亚洲欧美一区二区三区国产精品 | 在线日本中文字幕| 欧美久久成人|