成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Spring Cloud Hystrix的請求合并

開發 開發工具
通常微服務架構中的依賴通過遠程調用實現,而遠程調用中最常見的問題就是通信消耗與連接數占用。在高并發的情況之下,因通信次數的增加,總的通信時間消耗將會變的不那么理想。同時,因為對依賴服務的線程池資源有限,將出現排隊等待與響應延遲的情況。為了優化這兩個問題,Hystrix提供了HystrixCollapser來實現請求的合并,以減少通信消耗和線程數的占用。

[[211861]]

通常微服務架構中的依賴通過遠程調用實現,而遠程調用中最常見的問題就是通信消耗與連接數占用。在高并發的情況之下,因通信次數的增加,總的通信時間消耗將會變的不那么理想。同時,因為對依賴服務的線程池資源有限,將出現排隊等待與響應延遲的情況。為了優化這兩個問題,Hystrix提供了HystrixCollapser來實現請求的合并,以減少通信消耗和線程數的占用。

HystrixCollapser實現了在HystrixCommand之前放置一個合并處理器,它將處于一個很短時間窗(默認10毫秒)內對同一依賴服務的多個請求進行整合并以批量方式發起請求的功能(服務提供方也需要提供相應的批量實現接口)。通過HystrixCollapser的封裝,開發者不需要去關注線程合并的細節過程,只需要關注批量化服務和處理。下面我們從HystrixCollapser的使用實例,對其合并請求的過程一探究竟。

Hystrix的請求合并示例

  1. public abstract class HystrixCollapser<BatchReturnType, ResponseType, RequestArgumentType> implements  
  2.         HystrixExecutable<ResponseType>, HystrixObservable<ResponseType> { 
  3.     ... 
  4.     public abstract RequestArgumentType getRequestArgument(); 
  5.  
  6.     protected abstract HystrixCommand<BatchReturnType> createCommand(Collection<CollapsedRequest<ResponseType, RequestArgumentType>> requests); 
  7.  
  8.     protected abstract void mapResponseToRequests(BatchReturnType batchResponse, Collection<CollapsedRequest<ResponseType, RequestArgumentType>> requests); 
  9.     ... 

從HystrixCollapser抽象類的定義中可以看到,它指定了三個不同的類型:

  • BatchReturnType:合并后批量請求的返回類型
  • ResponseType:單個請求返回的類型
  • RequestArgumentType:請求參數類型

而對于這三個類型的使用可以在它的三個抽象方法中看到:

  • RequestArgumentType getRequestArgument():該函數用來定義獲取請求參數的方法。
  • HystrixCommand<BatchReturnType> createCommand(Collection<CollapsedRequest<ResponseType, RequestArgumentType>> requests):合并請求產生批量命令的具體實現方法。
  • mapResponseToRequests(BatchReturnType batchResponse, Collection<CollapsedRequest<ResponseType, RequestArgumentType>> requests):批量命令結果返回后的處理,這里需要實現將批量結果拆分并傳遞給合并前的各個原子請求命令的邏輯。

接下來,我們通過一個簡單的示例來直觀的理解實現請求合并的過程。

假設,當前微服務USER-SERVICE提供了兩個獲取User的接口:

  • /users/{id}:根據id返回User對象的GET請求接口。
  • /users?ids={ids}:根據ids參數返回User對象列表的GET請求接口,其中ids為以逗號分割的id集合。

而在服務消費端,為這兩個遠程接口已經通過RestTemplate實現了簡單的調用,具體如下:

  1. @Service 
  2. public class UserServiceImpl implements UserService { 
  3.  
  4.     @Autowired 
  5.     private RestTemplate restTemplate; 
  6.  
  7.     @Override 
  8.     public User find(Long id) { 
  9.         return restTemplate.getForObject("http://USER-SERVICE/users/{1}"User.class, id); 
  10.     } 
  11.  
  12.     @Override 
  13.     public List<User> findAll(List<Long> ids) { 
  14.         return restTemplate.getForObject("http://USER-SERVICE/users?ids={1}", List.class, StringUtils.join(ids, ",")); 
  15.     } 
  16.  

接著,我們來實現將短時間內多個獲取單一User對象的請求命令進行合并的實現:

  • ***步:為請求合并的實現準備一個批量請求命令的實現,具體如下:
  1. public class UserBatchCommand extends HystrixCommand<List<User>> { 
  2.  
  3.     UserService userService; 
  4.     List<Long> userIds; 
  5.  
  6.     public UserBatchCommand(UserService userService, List<Long> userIds) { 
  7.         super(Setter.withGroupKey(asKey("userServiceCommand"))); 
  8.         this.userIds = userIds; 
  9.         this.userService = userService; 
  10.     } 
  11.  
  12.     @Override 
  13.     protected List<User> run() throws Exception { 
  14.         return userService.findAll(userIds); 
  15.     } 
  16.  

批量請求命令實際上就是一個簡單的HystrixCommand實現,從上面的實現中可以看到它通過調用userService.findAll方法來訪問/users?ids={ids}接口以返回User的列表結果。

  • 第二步,通過繼承HystrixCollapser實現請求合并器:
  1. public class UserCollapseCommand extends HystrixCollapser<List<User>, User, Long> { 
  2.  
  3.     private UserService userService; 
  4.     private Long userId; 
  5.  
  6.     public UserCollapseCommand(UserService userService, Long userId) { 
  7.         super(Setter.withCollapserKey(HystrixCollapserKey.Factory.asKey("userCollapseCommand")).andCollapserPropertiesDefaults( 
  8.                 HystrixCollapserProperties.Setter().withTimerDelayInMilliseconds(100))); 
  9.         this.userService = userService; 
  10.         this.userId = userId; 
  11.     } 
  12.  
  13.     @Override 
  14.     public Long getRequestArgument() { 
  15.         return userId; 
  16.     } 
  17.  
  18.     @Override 
  19.     protected HystrixCommand<List<User>> createCommand(Collection<CollapsedRequest<User, Long>> collapsedRequests) { 
  20.         List<Long> userIds = new ArrayList<>(collapsedRequests.size()); 
  21.         userIds.addAll(collapsedRequests.stream().map(CollapsedRequest::getArgument).collect(Collectors.toList())); 
  22.         return new UserBatchCommand(userService, userIds); 
  23.     } 
  24.  
  25.     @Override 
  26.     protected void mapResponseToRequests(List<User> batchResponse, Collection<CollapsedRequest<User, Long>> collapsedRequests) { 
  27.         int count = 0; 
  28.         for (CollapsedRequest<User, Long> collapsedRequest : collapsedRequests) { 
  29.             User user = batchResponse.get(count++); 
  30.             collapsedRequest.setResponse(user); 
  31.         } 
  32.     } 
  33.  

在上面的構造函數中,我們為請求合并器設置了時間延遲屬性,合并器會在該時間窗內收集獲取單個User的請求并在時間窗結束時進行合并組裝成單個批量請求。下面getRequestArgument方法返回給定的單個請求參數userId,而createCommand和mapResponseToRequests是請求合并器的兩個核心:

  • createCommand:該方法的collapsedRequests參數中保存了延遲時間窗中收集到的所有獲取單個User的請求。通過獲取這些請求的參數來組織上面我們準備的批量請求命令
  • UserBatchCommand實例。

mapResponseToRequests:在批量命令UserBatchCommand實例被觸發執行完成之后,該方法開始執行,其中batchResponse參數保存了createCommand中組織的批量請求命令的返回結果,而collapsedRequests參數則代表了每個被合并的請求。在這里我們通過遍歷批量結果batchResponse對象,為collapsedRequests中每個合并前的單個請求設置返回結果,以此完成批量結果到單個請求結果的轉換。

請求合并的原理分析

下圖展示了在未使用HystrixCollapser請求合并器之前的線程使用情況??梢钥吹疆敺障M者同時對USER-SERVICE的/users/{id}接口發起了五個請求時,會向該依賴服務的獨立線程池中申請五個線程來完成各自的請求操作。

而在使用了HystrixCollapser請求合并器之后,相同情況下的線程占用如下圖所示。由于同一時間發生的五個請求處于請求合并器的一個時間窗內,這些發向/users/{id}接口的請求被請求合并器攔截下來,并在合并器中進行組合,然后將這些請求合并成一個請求發向USER-SERVICE的批量接口/users?ids={ids},在獲取到批量請求結果之后,通過請求合并器再將批量結果拆分并分配給每個被合并的請求。從圖中我們可以看到以來,通過使用請求合并器有效地減少了對線程池中資源的占用。所以在資源有效并且在短時間內會產生高并發請求的時候,為避免連接不夠用而引起的延遲可以考慮使用請求合并器的方式來處理和優化。

使用注解實現請求合并器

在快速入門的例子中,我們使用@HystrixCommand注解優雅地實現了HystrixCommand的定義,那么對于請求合并器是否也可以通過注解來定義呢?答案是肯定!

以上面實現的請求合并器為例,也可以通過如下方式實現:

  1. @Service 
  2. public class UserService { 
  3.  
  4.     @Autowired 
  5.     private RestTemplate restTemplate; 
  6.  
  7.     @HystrixCollapser(batchMethod = "findAll", collapserProperties = { 
  8.             @HystrixProperty(name="timerDelayInMilliseconds", value = "100"
  9.     }) 
  10.     public User find(Long id) { 
  11.         return null
  12.     } 
  13.  
  14.     @HystrixCommand 
  15.     public List<User> findAll(List<Long> ids) { 
  16.         return restTemplate.getForObject("http://USER-SERVICE/users?ids={1}", List.class, StringUtils.join(ids, ",")); 
  17.     } 

@HystrixCommand我們之前已經介紹過了,可以看到這里通過它定義了兩個Hystrix命令,一個用于請求/users/{id}接口,一個用于請求/users?ids={ids}接口。而在請求/users/{id}接口的方法上通過@HystrixCollapser注解為其創建了合并請求器,通過batchMethod屬性指定了批量請求的實現方法為findAll方法(即:請求/users?ids={ids}接口的命令),同時通過collapserProperties屬性為合并請求器設置相關屬性,這里使用@HystrixProperty(name="timerDelayInMilliseconds", value = "100")將合并時間窗設置為100毫秒。這樣通過@HystrixCollapser注解簡單而又優雅地實現了在/users/{id}依賴服務之前設置了一個批量請求合并器。

請求合并的額外開銷

雖然通過請求合并可以減少請求的數量以緩解依賴服務線程池的資源,但是在使用的時候也需要注意它所帶來的額外開銷:用于請求合并的延遲時間窗會使得依賴服務的請求延遲增高。比如:某個請求在不通過請求合并器訪問的平均耗時為5ms,請求合并的延遲時間窗為10ms(默認值),那么當該請求的設置了請求合并器之后,最壞情況下(在延遲時間窗結束時才發起請求)該請求需要15ms才能完成。

由于請求合并器的延遲時間窗會帶來額外開銷,所以我們是否使用請求合并器需要根據依賴服務調用的實際情況來選擇,主要考慮下面兩個方面:

  • 請求命令本身的延遲。如果依賴服務的請求命令本身是一個高延遲的命令,那么可以使用請求合并器,因為延遲時間窗的時間消耗就顯得莫不足道了。
  • 延遲時間窗內的并發量。如果一個時間窗內只有1-2個請求,那么這樣的依賴服務不適合使用請求合并器,這種情況下不但不能提升系統性能,反而會成為系統瓶頸,因為每個請求都需要多消耗一個時間窗才響應。相反,如果一個時間窗內具有很高的并發量,并且服務提供方也實現了批量處理接口,那么使用請求合并器可以有效的減少網絡連接數量并極大地提升系統吞吐量,此時延遲時間窗所增加的消耗就可以忽略不計了。

【本文為51CTO專欄作者“翟永超”的原創稿件,轉載請通過51CTO聯系作者獲取授權】

戳這里,看該作者更多好文

責任編輯:武曉燕 來源: 51CTO專欄
相關推薦

2023-02-03 15:16:42

SpringHystrix

2022-12-08 08:27:18

HystrixQPS數據

2022-09-15 15:25:47

spring-微服務

2017-07-03 09:50:07

Spring Clou微服務架構

2017-07-04 17:35:46

微服務架構Spring Clou

2021-11-16 11:45:00

SpringSpring ClouJava

2018-06-01 23:08:01

Spring Clou微服務服務器

2017-09-20 09:46:38

Spring BootSpring Clou內存

2023-11-26 00:42:07

微服務日志

2021-06-04 08:48:46

Spring ClouMaven Centr版本

2022-03-30 08:21:57

合并HTTP

2022-08-11 09:17:38

架構開發

2025-06-09 01:01:00

2024-05-20 09:19:45

請求合并容器

2022-05-26 00:00:00

網絡請求合并優化

2018-07-24 13:01:52

前端優化前端性能瀏覽器

2018-07-27 15:43:24

Spring Clou管理架構

2023-12-19 09:33:40

微服務監控

2022-06-09 08:30:59

Istiospring clo

2017-09-05 14:05:11

微服務spring clou路由
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 精品国产乱码久久久久久老虎 | 影音先锋中文字幕在线观看 | 午夜影院在线观看 | 欧美videosex性极品hd | 中文在线一区 | 日日骚网| 91精品国产91久久综合桃花 | 91中文字幕在线 | av免费网站在线观看 | 最新日韩av | 欧美激情精品久久久久久变态 | 国产情侣在线看 | 欧美色欧美亚洲另类七区 | 国产成人精品一区二区三 | 日韩成人一区二区 | 69热视频在线观看 | 久久机热| 在线观看国产视频 | 综合九九 | 成人国产综合 | 久久不卡视频 | 亚欧精品 | 久久久网 | 精品视频一区二区三区在线观看 | 中文字幕一区二区三区乱码图片 | 久久综合久色欧美综合狠狠 | 欧美精品成人一区二区三区四区 | 中文精品久久 | 台湾a级理论片在线观看 | 中文字幕免费 | 欧美a√ | 超碰操 | 99久久精品免费 | 爱爱视频在线观看 | 色婷婷在线视频 | 亚洲色在线视频 | 男人天堂网址 | 欧美日韩在线观看视频 | 欧美性网| 国产精品不卡 | 国产一级视频在线 |