Spring Cloud(三) 服務提供者 Eureka + 服務消費者(rest + Ribbon)

搜雲庫技術團隊發表於2017-12-06

Spring Cloud Ribbon是基於Netflix Ribbon實現的一套客戶端負載均衡的工具。它是一個基於HTTP和TCP的客戶端負載均衡器。它可以通過在客戶端中配置ribbonServerList來設定服務端列表去輪詢訪問以達到均衡負載的作用。

Ribbon是什麼?

Ribbon是Netflix釋出的開源專案,主要功能是提供客戶端的軟體負載均衡演算法,將Netflix的中間層服務連線在一起。Ribbon客戶端元件提供一系列完善的配置項如連線超時,重試等。簡單的說,就是在配置檔案中列出Load Balancer(簡稱LB)後面所有的機器,Ribbon會自動的幫助你基於某種規則(如簡單輪詢,隨即連線等)去連線這些機器。我們也很容易使用Ribbon實現自定義的負載均衡演算法。

LB方案分類

目前主流的LB方案可分成兩類:一種是集中式LB, 即在服務的消費方和提供方之間使用獨立的LB設施(可以是硬體,如F5, 也可以是軟體,如nginx), 由該設施負責把訪問請求通過某種策略轉發至服務的提供方;另一種是程式內LB,將LB邏輯整合到消費方,消費方從服務註冊中心獲知有哪些地址可用,然後自己再從這些地址中選擇出一個合適的伺服器。Ribbon就屬於後者,它只是一個類庫,整合於消費方程式,消費方通過它來獲取到服務提供方的地址。

Ribbon的主要元件與工作流程

微服務架構的核心思想是,一個應用是由多個小的、相互獨立的、微服務組成,這些服務執行在自己的程式中,開發和釋出都沒有依賴。 不同服務通過一些輕量級互動機制來通訊,例如 RPC、HTTP 等,服務可獨立擴充套件伸縮,每個服務定義了明確的邊界,不同的服務甚至可以採用不同的程式語言來實現,由獨立的團隊來維護。 簡單的來說,一個系統的不同模組轉變成不同的服務!而且服務可以使用不同的技術加以實現!

Ribbon的核心元件

均為介面型別,有以下幾個

ServerList

  • 用於獲取地址列表。它既可以是靜態的(提供一組固定的地址),也可以是動態的(從註冊中心中定期查詢地址列表)。

ServerListFilter

  • 僅當使用動態ServerList時使用,用於在原始的服務列表中使用一定策略過慮掉一部分地址。

IRule

  • 選擇一個最終的服務地址作為LB結果。選擇策略有輪詢、根據響應時間加權、斷路器(當Hystrix可用時)等。

Ribbon在工作時首選會通過ServerList來獲取所有可用的服務列表,然後通過ServerListFilter過慮掉一部分地址,最後在剩下的地址中通過IRule選擇出一臺伺服器作為最終結果。

Ribbon提供的主要負載均衡策略介紹

簡單輪詢負載均衡(RoundRobin)

以輪詢的方式依次將請求排程不同的伺服器,即每次排程執行i = (i + 1) mod n,並選出第i臺伺服器。

隨機負載均衡 (Random)

隨機選擇狀態為UP的Server

加權響應時間負載均衡 (WeightedResponseTime)

根據相應時間分配一個weight,相應時間越長,weight越小,被選中的可能性越低。

區域感知輪詢負載均衡(ZoneAvoidanceRule)

複合判斷server所在區域的效能和server的可用性選擇server

準備工作

本次專案示例,改造第一篇文章中的專案,使用spring-cloud-eureka-service作為服務註冊中心,spring-cloud-eureka-provider,複製三分,專案名稱依次修改為spring-cloud-eureka-provider-1 [1-3]

改造 Provider

服務提供者

在專案:spring-cloud-eureka-provider-1spring-cloud-eureka-provider-2spring-cloud-eureka-provider-3 的啟動類,都加入@Value("${server.port}"),修改home()方法, 來區分不同埠的Controller 響應,因為接下來,使用ribbon做均衡需要測試需要使用到

package io.ymq.example.eureka.provider;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@SpringBootApplication
@EnableEurekaClient
@RestController
public class EurekaProviderApplication {

    @Value("${server.port}")
    String port;

    @RequestMapping("/")
    public String home() {
        return "Hello world ,port:" + port;
    }

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

複製程式碼

修改配置

在專案:spring-cloud-eureka-provider-1spring-cloud-eureka-provider-2spring-cloud-eureka-provider-3,修改server: port:埠依次為8081,8082,8083

eureka:
  client:
    serviceUrl:
      defaultZone: http://localhost:8761/eureka/

spring:
  application:
    name: eureka-provider

server:
  port: 8081
複製程式碼

Ribbon Consumer

服務消費者

新增依賴

新建 spring-cloud-ribbon-consumer

<!-- 客戶端負載均衡 -->
<dependency>
	<groupId>org.springframework.cloud</groupId>
	<artifactId>spring-cloud-starter-ribbon</artifactId>
</dependency>

<!-- eureka客戶端 -->
<dependency>
	<groupId>org.springframework.cloud</groupId>
	<artifactId>spring-cloud-starter-eureka</artifactId>
</dependency>
複製程式碼

開啟服務負載均衡

在工程的啟動類中,通過@EnableDiscoveryClient向服務註冊中心註冊;並且向程式的ioc注入一個bean: restTemplate並通過@LoadBalanced註解表明這個restRemplate開啟負載均衡的功能。

package io.ymq.example.ribbon.consumer;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;
import org.springframework.cloud.client.loadbalancer.LoadBalanced;
import org.springframework.context.annotation.Bean;
import org.springframework.web.client.RestTemplate;

@EnableDiscoveryClient
@SpringBootApplication
public class RibbonConsumerApplication {

    @LoadBalanced
    @Bean
    RestTemplate restTemplate() {
        return new RestTemplate();
    }

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

複製程式碼

消費提供者方法

新建 ConsumerController 類,呼叫提供者的 hello 方法

package io.ymq.example.ribbon.consumer;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.client.RestTemplate;

/**
 * 描述:呼叫提供者的 `home` 方法
 *
 * @author yanpenglei
 * @create 2017-12-05 18:53
 **/
@RestController
public class ConsumerController {

    @Autowired
    private RestTemplate restTemplate;

    @GetMapping(value = "/hello")
    public String hello() {
        return restTemplate.getForEntity("http://eureka-provider/", String.class).getBody();
    }
}

複製程式碼

新增配置

完整配置 application.yml

指定服務的註冊中心地址,配置自己的服務埠,服務名稱

eureka:
  client:
    serviceUrl:
      defaultZone: http://localhost:8761/eureka/

spring:
  application:
    name: ribbon-consumer

server:
  port: 9000
複製程式碼

測試服務

啟動服務

依次啟動專案:

spring-cloud-eureka-service
spring-cloud-eureka-provider-1
spring-cloud-eureka-provider-2
spring-cloud-eureka-provider-3
spring-cloud-ribbon-consumer

啟動該工程後,訪問服務註冊中心,檢視服務是否都已註冊成功:http://localhost:8761/

檢視服務註冊狀態

負載均衡

在命令視窗curl http://localhost:9000/hello,發現Ribbon已經實現負載均衡

或者瀏覽器get 請求http://localhost:9000/hello F5 重新整理

測試Ribbon,負載均衡響應

原始碼下載

GitHub:github.com/souyunku/sp…

碼雲:gitee.com/souyunku/sp…

Contact

  • 作者:鵬磊
  • 出處:www.ymq.io
  • Email:admin@souyunku.com
  • 版權歸作者所有,轉載請註明出處
  • Wechat:關注公眾號,搜雲庫,專注於開發技術的研究與知識分享

關注公眾號-搜雲庫
搜雲庫

相關文章