成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Spring Cloud Gateway重試機(jī)制

shery / 1884人閱讀

摘要:還有就是重試次數(shù)如果太多的話會(huì)導(dǎo)致請求量加倍,給后端造成更大的壓力,設(shè)置合理的重試機(jī)制才是最關(guān)鍵的。今天我們來簡單的了解下中的重試機(jī)制和使用。使用講解是對請求重試提供的一個(gè)。

前言

重試,我相信大家并不陌生。在我們調(diào)用Http接口的時(shí)候,總會(huì)因?yàn)槟撤N原因調(diào)用失敗,這個(gè)時(shí)候我們可以通過重試的方式,來重新請求接口。

生活中這樣的事例很多,比如打電話,對方正在通話中啊,信號(hào)不好啊等等原因,你總會(huì)打不通,當(dāng)你第一次沒打通之后,你會(huì)打第二次,第三次...第四次就通了。

重試也要注意應(yīng)用場景,讀數(shù)據(jù)的接口比較適合重試的場景,寫數(shù)據(jù)的接口就需要注意接口的冪等性了。還有就是重試次數(shù)如果太多的話會(huì)導(dǎo)致請求量加倍,給后端造成更大的壓力,設(shè)置合理的重試機(jī)制才是最關(guān)鍵的。

今天我們來簡單的了解下Spring Cloud Gateway中的重試機(jī)制和使用。

使用講解

RetryGatewayFilter是Spring Cloud Gateway對請求重試提供的一個(gè)GatewayFilter Factory。

配置方式:

spring:
  cloud:
    gateway:
      routes:
      - id: fsh-house
        uri: lb://fsh-house
        predicates:
        - Path=/house/**
        filters:
        - name: Retry
          args:
           retries: 3
           series:
            - SERVER_ERROR
           statuses:
            - OK
           methods:
            - GET
            - POST
           exceptions:
            - java.io.IOException
配置講解

配置類源碼org.springframework.cloud.gateway.filter.factory.RetryGatewayFilterFactory.RetryConfig:

public static class RetryConfig {
    private int retries = 3;
        
    private List series = toList(Series.SERVER_ERROR);
        
    private List statuses = new ArrayList<>();
        
    private List methods = toList(HttpMethod.GET);

    private List> exceptions = toList(IOException.class);
        
    // .....
}

retries:重試次數(shù),默認(rèn)值是3次

series:狀態(tài)碼配置(分段),符合的某段狀態(tài)碼才會(huì)進(jìn)行重試邏輯,默認(rèn)值是SERVER_ERROR,值是5,也就是5XX(5開頭的狀態(tài)碼),共有5個(gè)值:

public enum Series {
    INFORMATIONAL(1),
    SUCCESSFUL(2),
    REDIRECTION(3),
    CLIENT_ERROR(4),
    SERVER_ERROR(5);
}

statuses:狀態(tài)碼配置,和series不同的是這邊是具體狀態(tài)碼的配置,取值請參考:org.springframework.http.HttpStatus

methods:指定哪些方法的請求需要進(jìn)行重試邏輯,默認(rèn)值是GET方法,取值如下:

public enum HttpMethod {
    GET, HEAD, POST, PUT, PATCH, DELETE, OPTIONS, TRACE;
}

exceptions:指定哪些異常需要進(jìn)行重試邏輯,默認(rèn)值是java.io.IOException

代碼測試

就寫個(gè)接口,在接口中記錄請求次數(shù),然后拋出一個(gè)異常模擬500,通過網(wǎng)關(guān)訪問這個(gè)接口,如果你配置了重試次數(shù)是3,那么接口中會(huì)輸出4次結(jié)果才是對的,證明重試生效了。

AtomicInteger ac = new AtomicInteger();

@GetMapping("/data")
public HouseInfo getData(@RequestParam("name") String name) {
    if (StringUtils.isBlank(name)) {
        throw new RuntimeException("error");
    }
    System.err.println(ac.addAndGet(1));
    return new HouseInfo(1L, "上海", "虹口", "XX小區(qū)");
}

更多Spring Cloud代碼盡在:https://github.com/yinjihuan/...

源碼欣賞
    @Override
    public GatewayFilter apply(RetryConfig retryConfig) {
        // 驗(yàn)證重試配置格式是否正確
        retryConfig.validate();

        Repeat statusCodeRepeat = null;
        if (!retryConfig.getStatuses().isEmpty() || !retryConfig.getSeries().isEmpty()) {
            Predicate> repeatPredicate = context -> {
                ServerWebExchange exchange = context.applicationContext();
                // 判斷重試次數(shù)是否已經(jīng)達(dá)到了配置的最大值
                if (exceedsMaxIterations(exchange, retryConfig)) {
                    return false;
                }
                // 獲取響應(yīng)的狀態(tài)碼
                HttpStatus statusCode = exchange.getResponse().getStatusCode();
                // 獲取請求方法類型
                HttpMethod httpMethod = exchange.getRequest().getMethod();
                // 判斷響應(yīng)狀態(tài)碼是否在配置中存在
                boolean retryableStatusCode = retryConfig.getStatuses().contains(statusCode);

                if (!retryableStatusCode && statusCode != null) { // null status code might mean a network exception?
                    // try the series
                    retryableStatusCode = retryConfig.getSeries().stream()
                            .anyMatch(series -> statusCode.series().equals(series));
                }
                // 判斷方法是否包含在配置中
                boolean retryableMethod = retryConfig.getMethods().contains(httpMethod);
                // 決定是否要進(jìn)行重試
                return retryableMethod && retryableStatusCode;
            };

            statusCodeRepeat = Repeat.onlyIf(repeatPredicate)
                    .doOnRepeat(context -> reset(context.applicationContext()));
        }

        //TODO: support timeout, backoff, jitter, etc... in Builder

        Retry exceptionRetry = null;
        if (!retryConfig.getExceptions().isEmpty()) {
            Predicate> retryContextPredicate = context -> {
                if (exceedsMaxIterations(context.applicationContext(), retryConfig)) {
                    return false;
                }
                // 異常判斷
                for (Class clazz : retryConfig.getExceptions()) {             
                    if (clazz.isInstance(context.exception())) {
                        return true;
                    }
                }
                return false;
            };
            // 使用reactor extra的retry組件
            exceptionRetry = Retry.onlyIf(retryContextPredicate)
                    .doOnRetry(context -> reset(context.applicationContext()))
                    .retryMax(retryConfig.getRetries());
        }


        return apply(statusCodeRepeat, exceptionRetry);
    }

    public boolean exceedsMaxIterations(ServerWebExchange exchange, RetryConfig retryConfig) {
        Integer iteration = exchange.getAttribute(RETRY_ITERATION_KEY);

        //TODO: deal with null iteration
        return iteration != null && iteration >= retryConfig.getRetries();
    }

    public void reset(ServerWebExchange exchange) {
        //TODO: what else to do to reset SWE?
        exchange.getAttributes().remove(ServerWebExchangeUtils.GATEWAY_ALREADY_ROUTED_ATTR);
    }

    public GatewayFilter apply(Repeat repeat, Retry retry) {
        return (exchange, chain) -> {
            if (log.isTraceEnabled()) {
                log.trace("Entering retry-filter");
            }

            // chain.filter returns a Mono
            Publisher publisher = chain.filter(exchange)
                    //.log("retry-filter", Level.INFO)
                    .doOnSuccessOrError((aVoid, throwable) -> {
                        // 獲取已經(jīng)重試的次數(shù),默認(rèn)值為-1
                        int iteration = exchange.getAttributeOrDefault(RETRY_ITERATION_KEY, -1);
                        // 增加重試次數(shù)
                        exchange.getAttributes().put(RETRY_ITERATION_KEY, iteration + 1);
                    });

            if (retry != null) {
                // retryWhen returns a Mono
                // retry needs to go before repeat
                publisher = ((Mono)publisher).retryWhen(retry.withApplicationContext(exchange));
            }
            if (repeat != null) {
                // repeatWhen returns a Flux
                // so this needs to be last and the variable a Publisher
                publisher = ((Mono)publisher).repeatWhen(repeat.withApplicationContext(exchange));
            }

            return Mono.fromDirect(publisher);
        };
    }
歡迎加入我的知識(shí)星球,一起交流技術(shù),免費(fèi)學(xué)習(xí)猿天地的課程(http://cxytiandi.com/course)

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/73649.html

相關(guān)文章

  • Spring Cloud Gateway實(shí)戰(zhàn)之五:內(nèi)置filter

    摘要:歡迎訪問我的歡迎訪問我的內(nèi)容所有原創(chuàng)文章分類匯總及配套源碼,涉及等本篇概覽本篇概覽作為實(shí)戰(zhàn)系列的第五篇,是時(shí)候了解過濾器的作用了,本篇咱們一起來了解內(nèi)置好的過濾器,真是種類繁多功能強(qiáng)大過濾器顧名思義,就是在請求頭部添加指定的內(nèi)容帶有的完整配歡迎訪問我的GitHubhttps://github.com/zq2599/blog_demos內(nèi)容:所有原創(chuàng)文章分類匯總及配套源碼,涉及Java、Doc...

    reclay 評論0 收藏0
  • SpringCloud升級(jí)之路2020.0.x版-41. SpringCloudGateway 基本

    摘要:將請求封裝成將請求封裝成的接口定義是但是最外層傳進(jìn)來的參數(shù)是和,需要將他們封裝成,這個(gè)工作就是在中做的。其實(shí)主要任務(wù)就是將各種參數(shù)封裝成除了和本次請求相關(guān)的和,還有會(huì)話管理器,編碼解碼器配置,國際化配置還有用于擴(kuò)展。本系列代碼地址:https://github.com/JoJoTec/spring-cloud-parent接下來,將進(jìn)入我們升級(jí)之路的又一大模塊,即網(wǎng)關(guān)模塊。網(wǎng)關(guān)模塊我們廢棄了...

    不知名網(wǎng)友 評論0 收藏0
  • springcloud-feign實(shí)現(xiàn)服務(wù)發(fā)現(xiàn)

    摘要:本文重點(diǎn)介紹一下基于實(shí)現(xiàn)服務(wù)發(fā)現(xiàn)。使用方式下面我們開始的使用添加和的依賴添加注解開啟服務(wù)發(fā)現(xiàn),注解支持客戶端。同樣子,他可以使用默認(rèn)的也可以使用或者修改配置文件服務(wù)名字服務(wù)無端口會(huì)隨機(jī)選擇一個(gè)服務(wù)集群名字注冊中心地址,完成。 springcloud-feign實(shí)現(xiàn)服務(wù)發(fā)現(xiàn) 上一篇介紹了nacos實(shí)現(xiàn)配置和注冊中心,在微服務(wù)中只有配置和注冊中心遠(yuǎn)遠(yuǎn)不夠,還需要有服務(wù)發(fā)現(xiàn)。本文重點(diǎn)介紹一...

    _DangJin 評論0 收藏0
  • spring cloud gateway 之限流篇

    摘要:常見的限流方式,比如適用線程池隔離,超過線程池的負(fù)載,走熔斷的邏輯。在令牌桶算法中,存在一個(gè)桶,用來存放固定數(shù)量的令牌。,令牌桶每秒填充平均速率。 轉(zhuǎn)載請標(biāo)明出處: https://www.fangzhipeng.com本文出自方志朋的博客 在高并發(fā)的系統(tǒng)中,往往需要在系統(tǒng)中做限流,一方面是為了防止大量的請求使服務(wù)器過載,導(dǎo)致服務(wù)不可用,另一方面是為了防止網(wǎng)絡(luò)攻擊。 常見的限流方式,...

    joy968 評論0 收藏0
  • SpringCloud(第 025 篇)Zuul 路由后面的微服務(wù)掛了后,Zuul 提供了一種回退機(jī)

    摘要:英文意思就是說提供一個(gè)回退機(jī)制當(dāng)路由后面的服務(wù)發(fā)生故障時(shí)。注意注解能注冊到服務(wù)上,是因?yàn)樵撟⒔獍丝蛻舳说淖⒔?,該是一個(gè)復(fù)合注解。地址可以查看該微服務(wù)網(wǎng)關(guān)代理了多少微服務(wù)的。 SpringCloud(第 025 篇)Zuul 路由后面的微服務(wù)掛了后,Zuul 提供了一種回退機(jī)制來應(yīng)對熔斷處理 - 一、大致介紹 1、在一些不穩(wěn)定因素導(dǎo)致路由后面的微服務(wù)宕機(jī)或者無響應(yīng)時(shí),zuul 就會(huì)累...

    Jochen 評論0 收藏0

發(fā)表評論

0條評論

shery

|高級(jí)講師

TA的文章

閱讀更多
最新活動(dòng)
閱讀需要支付1元查看
<