国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

RestTemplate集成Ribbbon

wall2flower / 3265人閱讀

摘要:的類圖如下主要根據創建擴展了,創建攔截的,這里會設置攔截器,這是集成的核心,當發起請求調用的時候,會先經過攔截器,然后才真正發起請求。和是配合使用的,最大重試次數是針對每一個的,如果設置,這樣觸發最大重試次數就是次。

上一篇文章我們分析了ribbon的核心原理,接下來我們來看看springcloud是如何集成ribbon的,不同的springcloud的組件(feign,zuul,RestTemplate)集成ribbon有所不同,這篇文章先來看看RestTemplate。

RestTemplate的類圖如下

HttpAccessor主要根據ClientHttpRequestFactory創建ClientHttpRequest

InterceptingHttpAccessor擴展了HttpAccessor,創建攔截的InterceptingClientHttpRequest,這里會設置攔截器ClientHttpRequestInterceptor,這是集成ribbon的核心,當RestTemplate發起http請求調用的時候,會先經過攔截器,然后才真正發起http請求。

攔截器ClientHttpRequestInterceptor是如何被設置的呢?在LoadBalancerAutoConfiguration類中,有如下代碼:

@LoadBalanced
@Autowired(required = false)
private List restTemplates = Collections.emptyList();

只要加入注解@LoadBalancedRestTemplate會被注入,在沒有引入spring retry組件的時候,加載如下配置:

@Configuration
@ConditionalOnMissingClass("org.springframework.retry.support.RetryTemplate")
static class LoadBalancerInterceptorConfig {
    @Bean
    public LoadBalancerInterceptor ribbonInterceptor(
        LoadBalancerClient loadBalancerClient,
        LoadBalancerRequestFactory requestFactory) {
        return new LoadBalancerInterceptor(loadBalancerClient, requestFactory);
    }

    @Bean
    @ConditionalOnMissingBean
    public RestTemplateCustomizer restTemplateCustomizer(
        final LoadBalancerInterceptor loadBalancerInterceptor) {
        return new RestTemplateCustomizer() {
            @Override
            public void customize(RestTemplate restTemplate) {
                List list = new ArrayList<>(
                    restTemplate.getInterceptors());
                list.add(loadBalancerInterceptor);
                restTemplate.setInterceptors(list);
            }
        };
    }
}

這樣RestTemplate就被設置了LoadBalancerInterceptor,下面來看看整個調用過程

整個過程有點復雜,核心就是經過攔截器LoadBalancerInterceptor,通過RibbonLoadBalancerClient發起負載均衡調用。RibbonLoadBalancerClientI組合了LoadBalancer,所以具備了負載均衡的能力,也就是我們在上一篇文章解讀的ribbon原理。

圖中我們沒有畫出真正發起http請求的過程,其默認是由SimpleClientHttpRequestFactory創建,ClientHttpRequestFactory的類圖如下:

從調用時序圖上我們看到,開始我們調用的是InterceptingClientHttpRequestFactory來獲取InterceptingClientHttpRequest,它們通過組合的方式集成了ClientHttpRequestFactory和攔截器,InterceptingClientHttpRequest發起調用的時候委托了其內部類InterceptingRequestExecution去處理,核心邏輯:

@Override
public ClientHttpResponse execute(HttpRequest request, byte[] body) throws IOException {
    if (this.iterator.hasNext()) {
        ClientHttpRequestInterceptor nextInterceptor = this.iterator.next();
        return nextInterceptor.intercept(request, body, this);
    }else {
        ClientHttpRequest delegate = requestFactory.createRequest(request.getURI(), request.getMethod());
        for (Map.Entry> entry : request.getHeaders().entrySet()) {
            List values = entry.getValue();
            for (String value : values) {
                delegate.getHeaders().add(entry.getKey(), value);
            }
        }
        if (body.length > 0) {
            StreamUtils.copy(body, delegate.getBody());
        }
        return delegate.execute();
    }
}

首先會先取出攔截器集合的第一個執行,當攔截器執行完成后,會回調回來,執行else的代碼,真正發起http請求,主要有兩種方式實現ClientHttpRequestFactory接口:

一種是SimpleClientHttpRequestFactory,使用J2SE提供的方式(既java.net包提供的方式)創建底層的Http請求連接

一種方式是使用HttpComponentsClientHttpRequestFactory方式,底層使用HttpClient訪問遠程的Http服務,使用HttpClient可以配置連接池和證書等信息。

?RestTemplate默認是使用SimpleClientHttpRequestFactory,內部是調用jdk的HttpConnection,默認超時為-1,可以這樣設置超時時間:

@Bean
@LoadBalanced
public RestTemplate restTemplate() {
    SimpleClientHttpRequestFactory factory  = new SimpleClientHttpRequestFactory();
    factory.setConnectTimeout(1000 * 2);//連接超時時間
    factory.setReadTimeout(1000 * 1);//讀超時時間
    return new RestTemplate(factory);
}

使用HttpComponentsClientHttpRequestFactory方式可以使用連接池(推薦) ,還可以設置重試策略(具體沒有研究過)

如果想開啟重試機制,我們可以引入spring的retry組件


    org.springframework.retry
    spring-retry
    版本號

這樣springcloud-ribbon就會加重如下配置:

@Configuration
@ConditionalOnClass(RetryTemplate.class)
public static class RetryAutoConfiguration {
    @Bean
    public RetryTemplate retryTemplate() {
        RetryTemplate template =  new RetryTemplate();
        template.setThrowLastExceptionOnExhausted(true);
        return template;
    }

    @Bean
    @ConditionalOnMissingBean
    public LoadBalancedRetryPolicyFactory loadBalancedRetryPolicyFactory() {
        return new LoadBalancedRetryPolicyFactory.NeverRetryFactory();
    }
}

@Configuration
@ConditionalOnClass(RetryTemplate.class)
public static class RetryInterceptorAutoConfiguration {
    @Bean
    @ConditionalOnMissingBean
    public RetryLoadBalancerInterceptor ribbonInterceptor(
        LoadBalancerClient loadBalancerClient, LoadBalancerRetryProperties properties,
        LoadBalancedRetryPolicyFactory lbRetryPolicyFactory,
        LoadBalancerRequestFactory requestFactory) {
        return new RetryLoadBalancerInterceptor(loadBalancerClient, properties,
                                                lbRetryPolicyFactory, requestFactory);
    }

    @Bean
    @ConditionalOnMissingBean
    public RestTemplateCustomizer restTemplateCustomizer(
        final RetryLoadBalancerInterceptor loadBalancerInterceptor) {
        return new RestTemplateCustomizer() {
            @Override
            public void customize(RestTemplate restTemplate) {
                List list = new ArrayList<>(
                    restTemplate.getInterceptors());
                list.add(loadBalancerInterceptor);
                restTemplate.setInterceptors(list);
            }
        };
    }
}
@Bean
@ConditionalOnClass(name = "org.springframework.retry.support.RetryTemplate")
@ConditionalOnMissingBean
    public LoadBalancedRetryPolicyFactory loadBalancedRetryPolicyFactory(SpringClientFactory clientFactory) {
    return new RibbonLoadBalancedRetryPolicyFactory(clientFactory);
}

攔截器替換成RetryLoadBalancerInterceptor了,這里集成了retry組件retryTemplate。重試策略由RetryHandler接口來配置,默認實現類DefaultLoadBalancerRetryHandler,如下為默認的配置參數

#最大的重試次數
ribbon.MaxAutoRetries=0
#最大重試server的個數
ribbon.MaxAutoRetriesNextServer=1
#是否開啟任何異常都重試(默認在get請求下會重試,其他情況不會重試,除非設置為true)
ribbon.OkToRetryOnAllOperations=false
#指定重試的http狀態碼
ribbon.retryableStatusCodes=500,501

以上是對全局生效,如果加上xxx.ribbon.MaxAutoRetries=1這樣只會對某個ribbon客戶端生效。MaxAutoRetries和MaxAutoRetriesNextServer是配合使用的,最大重試次數是針對每一個server的,如果設置MaxAutoRetries=1,MaxAutoRetriesNextServer=1這樣觸發最大重試次數就是4次。

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/76592.html

相關文章

  • SpringCloud(第 014 篇)電影 Ribbon 微服務集成 Hystrix 斷路器實現失

    摘要:當服務宕機或者不可用時,即請求超時后會調用此方法。添加電影微服務啟動類電影微服務集成斷路器實現失敗快速響應,達到熔斷效果。 SpringCloud(第 014 篇)電影 Ribbon 微服務集成 Hystrix 斷路器實現失敗快速響應,達到熔斷效果 - 一、大致介紹 1、Hystrix 斷路器的原理很簡單,如同電力過載保護器。它可以實現快速失敗,如果它在一段時間內偵測到許多類似的錯誤,...

    StonePanda 評論0 收藏0
  • SpringCloud(第 015 篇)電影Ribbon微服務集成Hystrix增加隔離策略控制指標

    摘要:傳播安全上下文或使用,通過增加的屬性,來增加相關的配置來達到執行隔離策略,控制線程數或者控制并發請求數來達到熔斷降級的作用。 SpringCloud(第 015 篇)電影Ribbon微服務集成Hystrix增加隔離策略控制線程數或請求數來達到熔斷降級的作用 - 一、大致介紹 1、本章節介紹關于Hystrix的2種隔離方式(Thread Pool 和 Semaphores); 2、Thr...

    RobinQu 評論0 收藏0
  • Spring Cloud中如何優雅的使用Feign調用接口

    摘要:對進行了封裝,使其支持標準注解和。可以與和組合使用以支持負載均衡。中使用當我們搭建好注冊中心之后,就是需要將自己的服務注冊到中,然后別的服務可以直接調用。 JAVA 項目中接口調用怎么做 ? Httpclient Okhttp Httpurlconnection RestTemplate 上面是最常見的幾種用法,我們今天要介紹的用法比上面的更簡單,方便,它就是 Feign Feig...

    ChanceWong 評論0 收藏0
  • Spring新功能

    摘要:新特性重要功能升級為了解決各種環境下如開發測試和生產選擇不同配置的問題,引入了環境功能。這個消息模塊支持的功能,同時提供了基于模板的方式發布消息是第一批支持特性的框架,比如它所支持的表達式。 Spring 3.1新特性 重要功能升級 為了解決各種環境下(如開發、測試和生產)選擇不同配置的問題,Spring 3.1引入了環境profile功能。借助于profile,就能根據應用部署在什...

    baiy 評論0 收藏0
  • SpringCloud學習(2)

    摘要:此為的結構圖上篇已注冊了,的服務,接下來用,實現負載均衡和的簡單客戶端,讓消費者調用服務。是發布的云中間層服務開源項目,其主要功能是提供客戶側軟件負載均衡算法,將的中間層服務連接在一起。對選定的負載均衡策略機上重試機制。 ??????????上篇已經搭建好基礎demo,接下來繼續構建項目并對spring cloud組件介紹描述。 showImg(https://segmentfault...

    wenzi 評論0 收藏0

發表評論

0條評論

最新活動
閱讀需要支付1元查看
<