欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Springcloud實(shí)現(xiàn)服務(wù)多版本控制的示例代碼

 更新時間:2020年05月15日 11:20:23   作者:Yuicon  
這篇文章主要介紹了Springcloud實(shí)現(xiàn)服務(wù)多版本控制的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

需求

小程序新版本上線需要審核,如果有接口新版本返回內(nèi)容發(fā)生了變化,后端直接上線會導(dǎo)致舊版本報錯,不上線審核又通不過。

之前是通過寫新接口來兼容,但是這樣會有很多兼容代碼或者冗余代碼,開發(fā)也不容易能想到這一點(diǎn),經(jīng)常直接修改了舊接口,于是版本控制就成了迫切的需求。

思路

所有請求都是走的網(wǎng)關(guān),很自然的就能想到在網(wǎng)關(guān)層實(shí)現(xiàn)版本控制。首先想到的是在ZuulFilter過濾器中實(shí)現(xiàn),前端所有請求都在請求頭中增加一個version的header,然后進(jìn)行匹配。但是這樣只能獲取到前端的版本,不能匹配選擇后端實(shí)例。

查詢資料后發(fā)現(xiàn)應(yīng)該在負(fù)載均衡的時候?qū)崿F(xiàn)版本控制。同樣是前端所有請求都在請求頭中增加一個version的header,后端實(shí)例都配置一個版本的tag。

實(shí)現(xiàn)

首先需要說明的是我選擇的控制中心是consul,網(wǎng)關(guān)是zuul。

負(fù)載均衡策略被抽象為IRule接口,項目默認(rèn)情況下使用的IRule的子類ZoneAvoidanceRule extends PredicateBasedRule,我們需要實(shí)現(xiàn)一個PredicateBasedRule的子類來替換ZoneAvoidanceRule。

PredicateBasedRule需要實(shí)現(xiàn)一個過濾的方法我們就在這個方法里實(shí)現(xiàn)版本控制,過濾后就是默認(rèn)的負(fù)載均衡策略了,默認(rèn)是輪詢。

  /**
   * Method that provides an instance of {@link AbstractServerPredicate} to be used by this class.
   * 
   */
  public abstract AbstractServerPredicate getPredicate();

VersionPredicate

我們可以看到PredicateBasedRule的getPredicate()方法需要返回一個AbstractServerPredicate實(shí)例,這個實(shí)例具體定義了版本控制的業(yè)務(wù)邏輯。代碼如下:

private static class VersionPredicate extends AbstractServerPredicate {

    private static final String VERSION_KEY = "version";

    @Override
    public boolean apply(@NullableDecl PredicateKey predicateKey) {
      if (predicateKey == null) {
        return true;
      }
      RequestContext ctx = RequestContext.getCurrentContext();
      HttpServletRequest request = ctx.getRequest();
      String version = request.getHeader(VERSION_KEY);
      if (version == null) {
        return true;
      }
      ConsulServer consulServer = (ConsulServer) predicateKey.getServer();
      if (!consulServer.getMetadata().containsKey(VERSION_KEY)) {
        return true;
      }
      return consulServer.getMetadata().get(VERSION_KEY).equals(version);
    }
  }

首先來了解下負(fù)載均衡的過程。一個請求到達(dá)網(wǎng)關(guān)后會解析出對應(yīng)的服務(wù)名,然后會獲取到該服務(wù)的所有可用實(shí)例,之后就會調(diào)用我們的過濾方法過濾出該請求可用的所有服務(wù)實(shí)例,最后進(jìn)行輪詢負(fù)載均衡。

PredicateKey類就是上層方法將可用實(shí)例Server和loadBalancerKey封裝后的類。版本控制的業(yè)務(wù)邏輯如下:

  • 判斷predicateKey是否為null,是的話直接返回true,true代表該實(shí)例可用
  • 通過RequestContext獲取當(dāng)前請求實(shí)例HttpServletRequest,再通過請求實(shí)例獲取請求頭里的版本號
  • 判斷前端請求是否帶了版本號,沒帶的話就不進(jìn)行版本控制直接返回true
  • 獲取服務(wù)實(shí)例并轉(zhuǎn)換成ConsulServer類,這里是因為我用的注冊中心是consul,選擇其他的可自行轉(zhuǎn)換成對應(yīng)的實(shí)現(xiàn)類
  • 判斷服務(wù)實(shí)例是否設(shè)置了版本號(例:spring.cloud.consul.discovery.tags="version=1.0.0"),可以看到我們是用consul的tags實(shí)現(xiàn)的版本控制,可以設(shè)置不同的tag實(shí)現(xiàn)很多功能
  • 同樣服務(wù)實(shí)例沒有設(shè)置版本號的話也是直接返回true
  • 最后進(jìn)行版本匹配,返回匹配成功的服務(wù)實(shí)例

注意的點(diǎn)

最終實(shí)現(xiàn)如下:

/**
 * @author Yuicon
 */
@Slf4j
public class VersionRule extends PredicateBasedRule {

  private final CompositePredicate predicate;

  public VersionRule() {
    super();
    this.predicate = createCompositePredicate(new VersionPredicate(),
        new AvailabilityPredicate(this, null));
  }

  @Override
  public AbstractServerPredicate getPredicate() {
    return this.predicate;
  }

  private CompositePredicate createCompositePredicate(VersionPredicate versionPredicate,
                            AvailabilityPredicate availabilityPredicate) {
    return CompositePredicate.withPredicates(versionPredicate, availabilityPredicate)
        .build();
  }

  private static class VersionPredicate extends AbstractServerPredicate {

    private static final String VERSION_KEY = "version";

    @Override
    public boolean apply(@NullableDecl PredicateKey predicateKey) {
      if (predicateKey == null) {
        return true;
      }
      RequestContext ctx = RequestContext.getCurrentContext();
      HttpServletRequest request = ctx.getRequest();
      String version = request.getHeader(VERSION_KEY);
      if (version == null) {
        return true;
      }
      ConsulServer consulServer = (ConsulServer) predicateKey.getServer();
      if (!consulServer.getMetadata().containsKey(VERSION_KEY)) {
        return true;
      }
      log.info("id is {}, header is {}, metadata is {}, result is {}",
          consulServer.getMetaInfo().getInstanceId(),
          version, consulServer.getMetadata().get(VERSION_KEY),
          consulServer.getMetadata().get(VERSION_KEY).equals(version));
      return consulServer.getMetadata().get(VERSION_KEY).equals(version);
    }
  }
}

原本我是加上@Component注解后在本地直接測試通過了。可是在更新到生產(chǎn)服務(wù)器后卻出現(xiàn)大部分請求都找不到的服務(wù)實(shí)例的錯誤,搞的我一頭霧水,趕緊回滾到原來的版本。

查詢了很多資料后才找到一篇文章,發(fā)現(xiàn)需要一個Config類來聲明替換原有的負(fù)載均衡策略類。代碼如下:

@RibbonClients(defaultConfiguration = RibbonGatewayConfig.class)
@Configuration
public class RibbonGatewayConfig {

  @Bean
  public IRule versionRule() {
    return new VersionRule();
  }
}

到此為止版本控制算是實(shí)現(xiàn)成功了。

結(jié)尾

在實(shí)際使用過程中發(fā)現(xiàn)還是有很多問題。比如前端版本號是全局唯一的,當(dāng)其中一個服務(wù)升級了版本號,就需要將所有服務(wù)都升級到該版本號,即使代碼沒有任何更改。比較好的解決方案是前端根據(jù)不同服務(wù)傳遞不同的版本號,不過前端反饋實(shí)現(xiàn)困難。

還有個妥協(xié)的方案,就是利用配置中心來對具體服務(wù)是否開啟版本控制進(jìn)行配置,因為現(xiàn)在的需求只是一小段時間里需要版本控制,小程序?qū)徍诉^后就可以把舊服務(wù)實(shí)例關(guān)了。大家如果有更好的方案歡迎討論。

到此這篇關(guān)于Springcloud實(shí)現(xiàn)服務(wù)多版本控制的示例代碼的文章就介紹到這了,更多相關(guān)Springcloud多版本控制內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 使用Java填充Word模板的方法詳解

    使用Java填充Word模板的方法詳解

    Java填充Word模板是一種將動態(tài)數(shù)據(jù)插入到Word文檔模板中生成最終文檔的過程,通常用于批量創(chuàng)建包含個人信息、報告結(jié)果或其他動態(tài)內(nèi)容的文檔,本文給大家介紹了使用Java填充Word模板的方法,需要的朋友可以參考下
    2024-07-07
  • RocketMQ事務(wù)消息保證消息的可靠性和一致性

    RocketMQ事務(wù)消息保證消息的可靠性和一致性

    RocketMQ事務(wù)消息是一種能夠保證消息傳遞的可靠性和一致性的消息傳遞模式。它通過引入“半消息”和“事務(wù)狀態(tài)”機(jī)制,實(shí)現(xiàn)了消息發(fā)送和本地事務(wù)執(zhí)行的原子性,從而確保了消息的可靠性和一致性
    2023-04-04
  • seata的部署和集成詳細(xì)介紹

    seata的部署和集成詳細(xì)介紹

    這篇文章主要介紹了Java seata的部署和集成,文章中有詳細(xì)的代碼示例和圖片講解,對學(xué)習(xí)seata有一定的參考價值,需要的朋友可以參考一下
    2023-04-04
  • 在Mybatis @Select注解中實(shí)現(xiàn)拼寫動態(tài)sql

    在Mybatis @Select注解中實(shí)現(xiàn)拼寫動態(tài)sql

    這篇文章主要介紹了在Mybatis @Select注解中實(shí)現(xiàn)拼寫動態(tài)sql,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-11-11
  • 排序算法的Java實(shí)現(xiàn)全攻略

    排序算法的Java實(shí)現(xiàn)全攻略

    這篇文章主要介紹了排序算法的Java實(shí)現(xiàn),包括Collections.sort()的使用以及各種經(jīng)典算法的Java代碼實(shí)現(xiàn)方法總結(jié),超級推薦!需要的朋友可以參考下
    2015-08-08
  • Spring中IOC和AOP的核心組成架構(gòu)詳解

    Spring中IOC和AOP的核心組成架構(gòu)詳解

    這篇文章主要介紹了Spring中IOC和AOP的核心組成架構(gòu)詳解,本文是對Spring的2大核心功能——IoC和AOP 的總結(jié)提煉,并增加了環(huán)境profile和條件化bean的內(nèi)容,篇幅較短,更像是一個大綱,或者思維導(dǎo)圖,需要的朋友可以參考下
    2023-08-08
  • 詳解SpringBoot如何實(shí)現(xiàn)統(tǒng)一后端返回格式

    詳解SpringBoot如何實(shí)現(xiàn)統(tǒng)一后端返回格式

    在前后端分離的項目中后端返回的格式一定要友好,不然會對前端的開發(fā)人員帶來很多的工作量。那么SpringBoot如何做到統(tǒng)一的后端返回格式呢?本文將為大家詳細(xì)講講
    2022-04-04
  • mybatis plus自動生成代碼tinyint(1)自動轉(zhuǎn)換為Boolean的問題及解決

    mybatis plus自動生成代碼tinyint(1)自動轉(zhuǎn)換為Boolean的問題及解決

    這篇文章主要介紹了mybatis plus自動生成代碼tinyint(1)自動轉(zhuǎn)換為Boolean的問題及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-08-08
  • 身份證號碼驗證算法深入研究和Java實(shí)現(xiàn)

    身份證號碼驗證算法深入研究和Java實(shí)現(xiàn)

    這篇文章主要介紹了身份證號碼驗證算法深入研究和Java實(shí)現(xiàn),本文講解了18身份證號碼的結(jié)構(gòu)、根據(jù)17位數(shù)字本體碼獲取最后一位校驗碼程序?qū)嵗葍?nèi)容,需要的朋友可以參考下
    2015-06-06
  • Java實(shí)現(xiàn)的分頁工具類與用法示例

    Java實(shí)現(xiàn)的分頁工具類與用法示例

    這篇文章主要介紹了Java實(shí)現(xiàn)的分頁工具類與用法,結(jié)合完整實(shí)例形式分析了java分頁工具類的定義、使用方法及相關(guān)操作技巧,需要的朋友可以參考下
    2019-10-10

最新評論