SpringCloud+Eureka+Feign+Ribbon的簡化搭建流程,加入熔斷,網關和Redis緩存[2]

来源:https://www.cnblogs.com/jsccc520/archive/2019/12/30/12117304.html
-Advertisement-
Play Games

SpringCloud+Eureka+Feign+Ribbon的簡化搭建流程,加入熔斷,網關和Redis緩存 ...


作者:故事我忘了
個人微信公眾號:程式猿的月光寶盒

前提:本篇是基於

SpringCloud+Eureka+Feign+Ribbon的簡化搭建流程和CRUD練習[1]

的修改與拓展


1.修改consumerCenterFeign.java,把返回值全部設置為String

/**
 * 是consumer調用provider(需要指定provider的名字)
 * 請求的清單列表:規定調用地址、參數、返回值
 * 在正常走通的時候不走CenterFeignFallBack,當provider down的時候走熔斷器,相當於是類的try-catch
 */
@FeignClient(name = "provider",fallback = CenterFeignFallBack.class)
public interface CenterFeign {
    @GetMapping("/optionData.do")
    public String optionData();

    @PostMapping("/showEmpData.do")
    //Feign:不支持對象傳參,所以要用@RequestBody
    public String showEmpData(@RequestBody Emp emp);

    @PostMapping("/add.do")
    public String add(@RequestBody Emp emp);

    @PostMapping("/edit.do")
    public String edit(@RequestBody  Emp emp);

    @GetMapping("/del.do")
    public String del(@RequestParam("empno") Integer empno);
}

2.在CenterFeign.java的同包下創建實現了CenterFeign介面的CenterFeignFallBack.java作為熔斷器

@Component
public class CenterFeignFallBack implements CenterFeign {
    @Override
    public String optionData() {
        return "provider-optionData-error";
    }

    @Override
    public String showEmpData(Emp emp) {
        return "provider-showEmpData-error";
    }

    @Override
    public String add(Emp emp) {
        return "provider-add-error";
    }

    @Override
    public String edit(Emp emp) {
        return "provider-edit-error";
    }

    @Override
    public String del(Integer empno) {
        return "provider-del-error";
    }
}

3.修改consumerCenterController.java,返回值全改為String

@RestController
public class CenterController{
    @Resource
    private CenterFeign centerFeign;

    @GetMapping("/optionData-consumer.do")
    public String optionData() {
        return centerFeign.optionData();
    }

    @PostMapping("/showEmpData-consumer.do")
    public String showEmpData(@RequestBody Emp emp) {
        return centerFeign.showEmpData(emp);
    }

    @PostMapping("/add-consumer.do")
    public String add(@RequestBody Emp emp) {
        return centerFeign.add(emp);
    }

    @PostMapping("/edit-consumer.do")
    public String edit(@RequestBody Emp emp) {
        return centerFeign.edit(emp);
    }

    @GetMapping("/del-consumer.do")
    public String del(@RequestParam("empno") Integer empno) {
        return centerFeign.del(empno);
    }
}

4.確認consumer中的配置文件application.properties中有沒有開啟熔斷feign.hystrix.enabled=true,沒有的話加上

#埠號
server.port=8764
#應用名
spring.application.name=consumer
#eureka客戶端服務url預設區域
eureka.client.serviceUrl.defaultZone=http://localhost:8761/eureka/
#開啟熔斷器
feign.hystrix.enabled=true
#下線名稱.ribbon.NF載入平衡規則類名,這裡先註釋
provider.ribbon.NFLoadBalancerRuleClassName=com.netflix.loadbalancer.RandomRule

5.開啟服務測試

訪問http://localhost:8761/

發現所有服務已經註冊

圖片

正常走provider沒有問題

圖片


圖片

圖片

走consumer也沒有問題,只是返回的格式是字元串類型,不是json類型,但是沒關係,格式還是json的格式,前臺該怎麼轉還能怎麼轉,不影響

圖片

那怎麼走容錯?

現在手動停止一個provider

圖片

因為這裡我開了負載均衡策略(在consumer中的配置文件中provider.ribbon.NFLoadBalancerRuleClassName=com.netflix.loadbalancer.RandomRule),所以有一定幾率觸發熔斷器,

圖片

這就相當於類之間的try-catch,沒有熔斷器的話這裡百分百是報錯誤代碼.那這裡我先把負載均衡關掉,在測試有沒有走容錯,(猜一下,會走嗎?)

圖片

圖片

這裡我測了這麼多次,都沒有走熔斷,所以顯然不走,為何?

因為沒有手動設置負載均衡策略的話,預設走的是輪詢.機制,啥是Ribbon輪詢機制?

簡單的說就是有abc三台伺服器,正常的情況下走a->b->c,在有一臺down了的時候,那台就自動跳過,比如b down了,那麼就是a->c

以上,個人理解,若有誤請指正,よろしくお願いします~


現在,既然一臺伺服器工作是沒有問題 那我兩台provider全部停止呢?

圖片

那答案是肯定的,絕壁走容錯~~

圖片

6.測試完沒有問題,現在添加網關功能模塊

新建模塊(Spring initializr)zuul-client

圖片

圖片

7.編輯配置文件application.properties

#網關埠
server.port=8765
#應用名
spring.application.name=zuul
#網關路徑提供者,後面的**表示:xxx.do
zuul.routes.provider=/pro/**/
#網關路徑消費者,後面的**表示:xxx.do
zuul.routes.consumer=/con/**/

8.在啟動類上添加註解

//開啟網關代理
@EnableZuulProxy
//開啟eureka客戶端
@EnableEurekaClient
@SpringBootApplication
public class ZuulClientApplication {

    public static void main(String[] args) {
        SpringApplication.run(ZuulClientApplication.class, args);
    }

}

9.開啟服務測試

圖片


測試provider

圖片

測試consumer

圖片

10.網關作用

統一了所有客戶端的ip地址和埠號,我們只要給不同層級的應用起別名就ok了(就是這裡的圖片)

11.加入Redis緩存

11.1在provider-one模塊的pom文件中加入redis依賴

 <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

11.2修改provider-one模塊的DeptServiceImpl文件,加入@Autowired private RedisTemplate redisTemplate;

@Service
public class DeptServiceImpl implements DeptService {
    @Autowired
    private DeptMapper deptMapper;
    @Autowired
    private RedisTemplate redisTemplate;

    @Override
    public Map<String, Object> optionData() {
        Map<String, Object> map=new HashMap<>();
        List<Map<String, Object>> list = deptMapper.selAllDeptData();
        // 定義Redis存儲集合的對象
        ListOperations<String,Map<String,Object>> redisList = redisTemplate.opsForList();
        //拼接Redis中存儲數據對應的key
        String key = "depts";
        //判斷Redis中是否有key,沒有就說明是第一次訪問,將數據放入Redis
        if(!redisTemplate.hasKey(key)){
            //直接將資料庫查詢出來的值放入Redis
            System.out.println("provider-one-optionData的值已經放入Redis");
            redisList.leftPushAll(key,list);
        }
        map.put("data",list);
        return map;
    }
}

11.3對應的 修改DeptService,返回值變成map

public interface DeptService {
    Map<String,Object> optionData();
}

11.4修改CenterController,把返回值類型改為Map

    public Map<String, Object> optionData() {
        return deptService.optionData();
    }

11.5同樣的在provider-two的pom.xml中加入redis依賴

<dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

11.6修改provider-two模塊的DeptServiceImpl文件,加入@Autowired private RedisTemplate redisTemplate;

@Service
public class DeptServiceImpl implements DeptService {
    @Autowired
    private DeptMapper deptMapper;
    @Autowired
    private RedisTemplate redisTemplate;

    @Override
    public Map<String, Object> optionData() {
        Map<String, Object> map=new HashMap<>();
        List<Map<String, Object>> list = deptMapper.selAllDeptData();
        // 定義Redis存儲集合的對象
        ListOperations<String,Map<String,Object>> redisList = redisTemplate.opsForList();
        //拼接Redis中存儲數據對應的key
        String key = "depts";
        //判斷Redis中是否有key,沒有就說明是第一次訪問,將數據放入Redis
        if(!redisTemplate.hasKey(key)){
            //直接將資料庫查詢出來的值放入Redis
            System.out.println("provider-two-optionData的值已經放入Redis");
            redisList.leftPushAll(key,list);
        }
        map.put("data",list);
        return map;
    }
}

11.7對應的 修改DeptService,返回值變成map

public interface DeptService {
    Map<String,Object> optionData();
}

11.8修改CenterController,把返回值類型改為Map

    public Map<String, Object> optionData() {
        return deptService.optionData();
    }

11.9更新provider-one模塊的配置文件application.properties,加入Redis配置

#服務埠號
server.port=8762
#應用名
spring.application.name=provider
#eureka客戶端服務url預設區域
eureka.client.serviceUrl.defaultZone=http://localhost:8761/eureka/

#數據源驅動類名
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
#數據源url
spring.datasource.url=jdbc:mysql:///kh75
#數據源用戶名
spring.datasource.username=root
#數據源密碼
spring.datasource.password=admin



#後期會寫mapper.xml,這裡先註釋
#mybatis.mapper-locations=classpath:mapper/*.xml

#給實體類起別名,同樣這裡先註釋
#mybatis.type-aliases-package=cn.kgc.vo

#配置Redis
spring.redis.port=6379
#redis所在的主機地址
spring.redis.host=xxx.xxx.xxx
spring.redis.database=0

11.10更新provider-two模塊的配置文件application.properties,加入Redis配置

#服務埠號
server.port=8763
#應用名
spring.application.name=provider
#eureka客戶端服務url預設區域
eureka.client.serviceUrl.defaultZone=http://localhost:8761/eureka/

#數據源驅動類名
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
#數據源url
spring.datasource.url=jdbc:mysql:///kh75
#數據源用戶名
spring.datasource.username=root
#數據源密碼
spring.datasource.password=admin



#後期會寫mapper.xml,這裡先註釋
#mybatis.mapper-locations=classpath:mapper/*.xml

#給實體類起別名,同樣這裡先註釋
#mybatis.type-aliases-package=cn.kgc.vo
#配置Redis
spring.redis.port=6379
#redis所在的主機地址
spring.redis.host=xxx.xxx.xx
spring.redis.database=0

11.11更新consumer的CenterFeign,所有返回值都是map

@FeignClient(name = "provider",fallback = CenterFeignFallBack.class)
public interface CenterFeign {
    @GetMapping("/optionData.do")
     Map<String,Object> optionData();

    @PostMapping("/showEmpData.do")
    //Feign:不支持對象傳參,所以要用@RequestBody
    Map<String,Object> showEmpData(@RequestBody Emp emp);

    @PostMapping("/add.do")
    Map<String,Object> add(@RequestBody Emp emp);

    @PostMapping("/edit.do")
    Map<String,Object> edit(@RequestBody  Emp emp);

    @GetMapping("/del.do")
    Map<String,Object> del(@RequestParam("empno") Integer empno);
}

11.12更新CenterFeignFallBack

@Component
public class CenterFeignFallBack implements CenterFeign {
    private Map<String, Object> map = new HashMap<> ();
    @Autowired
    private RedisTemplate redisTemplate;
    @Override
    public Map<String,Object> optionData() {
         ListOperations<String,Map<String,Object>> redisList = redisTemplate.opsForList();
        map.put("msg","provider-optionData-error");
        map.put("feign-data",redisList.range("depts",0,-1));
        return map;
    }

    @Override
    public Map<String,Object> showEmpData(Emp emp) {
         ListOperations<String,Map<String,Object>> redisList = redisTemplate.opsForList();
        map.put("msg","provider-showEmpData-error");
        //這裡對應的serviceImp裡面還沒修改成Redis,先放著
        map.put("feign-data",redisList.range("emps",0,-1));
        return map;
    }

    @Override
    public Map<String,Object> add(Emp emp) {
        map.put("msg","provider-add-error");
        return map;
    }

    @Override
    public Map<String,Object> edit(Emp emp) {
        map.put("msg","provider-edit-error");
        return map;
    }

    @Override
    public Map<String,Object> del(Integer empno) {
        map.put("msg","provider-del-error");
        return map;
    }
}

11.13再修改consumer中的CenterController,返回值也全改成Map

@RestController
public class CenterController{
    @Resource
    private CenterFeign centerFeign;

    @GetMapping("/optionData-consumer.do")
    public Map<String,Object> optionData() {
        return centerFeign.optionData();
    }

    @PostMapping("/showEmpData-consumer.do")
    public Map<String,Object> showEmpData(@RequestBody Emp emp) {
        return centerFeign.showEmpData(emp);
    }

    @PostMapping("/add-consumer.do")
    public Map<String,Object> add(@RequestBody Emp emp) {
        return centerFeign.add(emp);
    }

    @PostMapping("/edit-consumer.do")
    public Map<String,Object> edit(@RequestBody Emp emp) {
        return centerFeign.edit(emp);
    }

    @GetMapping("/del-consumer.do")
    public Map<String,Object> del(@RequestParam("empno") Integer empno) {
        return centerFeign.del(empno);
    }
}

11.14檢查consumer的pom文件里是否有加入Redis依賴,沒有加上...

11.15 打開所有模塊進行測試

圖片

圖片

圖片

圖片

圖片

圖片

把提供者斷掉,理論上走熔斷和redis數據

圖片

圖片

以上....


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • axios 是一個輕量的 HTTP客戶端,它基於 XMLHttpRequest 服務來執行 HTTP 請求,支持豐富的配置,支持 Promise,支持瀏覽器端和 Node.js 端。在真實項目中為了提高我們的代碼質量,我們通常會對 axios 二次封裝一下再使用。這篇文章就帶你從零開始封裝 axio... ...
  • float實現文字環繞圖片效果: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>float</title> <style> body{ font-family: '微軟雅黑'; } .per{ width: ...
  • border-style的值: none 無 dotted 點狀 dashed 虛線 solid 實線 double 雙實線 margin: 垂直方向兩個相鄰元素都設置了外邊距,那麼外邊距會發生合併 合併高度=兩個發生合併的外邊距中的較大值 元素的實際高度=上邊框+上內邊距+內容高度+下內邊距+下邊 ...
  • 1、計算當前周一到周日的日期 ...
  • 1+x證書Web 前端開發初級——理論考試(試卷1) 一、單選題(每小題 2 分,共 30 小題,共 60 分) 1.HTML 語言中,設置表格中文字與邊框距離的標簽是() A、<table boder=””> B、<table cellspacing=””> C、<table cellpaddin ...
  • 對事件作出反應 JavaScript 能夠在事件發生時執行,比如當用戶點擊某個 HTML 元素時。 為了在用戶點擊元素時執行代碼,請向 HTML 事件屬性添加 JavaScript 代碼: HTML 事件的例子: 當用戶點擊滑鼠時 當網頁載入後 當圖像載入後 當滑鼠移至元素上時 當輸入欄位被改變時 ...
  • 通過Java日期時間API系列7 Jdk8中java.time包中的新的日期時間API類的優點,java8具有很多優點,現在網上查到的農曆轉換工具類都是基於jdk7及以前的類寫的,下麵使用java新的日期時間API重寫農曆LunarDate。 package com.xkzhangsan.time; ...
  • 1.到官網下載scala源代碼 點擊如下鏈接下載源碼:http://www.scala-lang.org/download/all.html 選擇需要的版本點擊進行下載,我選擇的是2.11.8版本,如下圖: 2.在idea中設置指向源代碼 在intellij中選擇 File –> Project S ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...