SpringCloud之Ribbon:負載均衡

Spring Cloud集成了Ribbon,結合Eureka,可實現客戶端的負載均衡。java

下面實現一個例子,結構下圖所示。web

1、服務器端spring

一、建立項目apache

開發工具:IntelliJ IDEA 2019.2.3
IDEA中建立一個新的SpringBoot項目,名稱爲「cloud-server」,SpringBoot版本選擇2.1.10,在選擇Dependencies(依賴)的界面勾選Spring Cloud Discovert ->
Eureka Server,建立完成後的pom.xml配置文件自動添加SpringCloud最新穩定版本依賴,當前爲Greenwich.SR3。
pom.xml完整內容以下:json

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.1.10.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.example</groupId>
    <artifactId>cloud-server</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>cloud-server</name>
    <description>Demo project for Spring Boot</description>

    <properties>
        <java.version>1.8</java.version>
        <spring-cloud.version>Greenwich.SR3</spring-cloud.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>${spring-cloud.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

</project>
View Code

二、修改配置application.yml瀏覽器

server:
  port: 8761
eureka:
  client:
    register-with-eureka: false
    fetch-registry: false

三、修改啓動類代碼CloudServerApplication.java服務器

增長註解@EnableEurekaServerapp

package com.example.cloudserver;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.server.EnableEurekaServer;

@SpringBootApplication
@EnableEurekaServer
public class CloudServerApplication {

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

}

2、服務提供者負載均衡

一、建立項目maven

IDEA中建立一個新的SpringBoot項目,除了名稱爲「cloud-provider」,其它步驟和上面建立服務器端同樣。

二、修改配置application.yml

spring:
  application:
    name: cloud-provider
eureka:
  instance:
    hostname: localhost
  client:
    serviceUrl:
      defaultZone: http://localhost:8761/eureka/

三、修改啓動類代碼CloudProviderApplication.java

增長註解@EnableEurekaClient;
讓類在啓動時讀取控制檯輸入,決定使用哪一個端口啓動服務器;
增長一個測試用的控制器方法。

package com.example.cloudprovider;

//import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.builder.SpringApplicationBuilder;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import javax.servlet.http.HttpServletRequest;
import java.util.Scanner;

@SpringBootApplication
@EnableEurekaClient
@RestController
public class CloudProviderApplication {

    public static void main(String[] args) {
        //SpringApplication.run(CloudProviderApplication.class, args);
        Scanner scan = new Scanner(System.in);
        String port = scan.nextLine();
        new SpringApplicationBuilder(CloudProviderApplication.class).properties("server.port=" + port).run(args);
    }

    @RequestMapping("/")
    public String index(HttpServletRequest request) {
        return request.getRequestURL().toString();
    }
}

3、服務調用者

一、建立項目
IDEA中建立一個新的SpringBoot項目,除了名稱爲「cloud-invoker」,其它步驟和上面建立服務器端同樣。

二、修改配置application.yml

server:
  port: 9000
spring:
  application:
    name: cloud-invoker
eureka:
  instance:
    hostname: localhost
  client:
    serviceUrl:
      defaultZone: http://localhost:8761/eureka/

三、修改啓動類代碼CloudInvokerApplication.java

增長註解@EnableDiscoveryClient。

package com.example.cloudinvoker;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;

@SpringBootApplication
@EnableDiscoveryClient
public class CloudInvokerApplication {

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

}

四、配置Ribbon有2種方式:使用代碼、使用配置文件

方式一:使用代碼
(1)新建一個自定義負載規則類MyRule.java
Ribbon的負載均衡器接口定義了服務器的操做,主要是用於進行服務器選擇。
調用ILoadBalancer的getAllServers方法能夠返回所有服務器,這裏只返回第一個服務器。

package com.example.cloudinvoker;

import com.netflix.loadbalancer.ILoadBalancer;
import com.netflix.loadbalancer.IRule;
import com.netflix.loadbalancer.Server;

import java.util.List;

public class MyRule implements IRule {

    private ILoadBalancer iLoadBalancer;

    @Override
    public Server choose(Object o) {
        List<Server> servers = iLoadBalancer.getAllServers();
        System.out.println("自定義服務器規則類,輸出服務器信息:");
        for(Server s: servers){
            System.out.println(" " + s.getHostPort());
        }
        return servers.get(0);
    }

    @Override
    public void setLoadBalancer(ILoadBalancer iLoadBalancer) {
        this.iLoadBalancer = iLoadBalancer;
    }

    @Override
    public ILoadBalancer getLoadBalancer() {
        return this.iLoadBalancer;
    }
}

(2)新建一個Ping類MyPing.java

負載均衡器中提供了Ping機制,每隔一段時間去Ping服務器,判斷服務器是否存活。
該工做由IPing接口的實現類負責。

package com.example.cloudinvoker;

import com.netflix.loadbalancer.IPing;
import com.netflix.loadbalancer.Server;

public class MyPing implements IPing {

    @Override
    public boolean isAlive(Server server) {
        System.out.println("自定義Ping類,服務器信息:" + server.getHostPort() + ",狀態:" + server.isAlive());
        return true;
    }
}

(3)新建配置類MyConfig.java

package com.example.cloudinvoker.config;

import com.example.cloudinvoker.MyPing;
import com.example.cloudinvoker.MyRule;
import com.netflix.loadbalancer.IPing;
import com.netflix.loadbalancer.IRule;
import org.springframework.context.annotation.Bean;

public class MyConfig {
    @Bean
    public IRule getRule(){
        return new MyRule();
    }
    @Bean
    public IPing getPing(){
        return new MyPing();
    }
}

(4)新建配置類CloudProviderConfig.java

package com.example.cloudinvoker.config;

import org.springframework.cloud.netflix.ribbon.RibbonClient;

@RibbonClient(name = "cloud-provider", configuration = MyConfig.class)
public class CloudProviderConfig {
}

方式二:使用配置文件

把方式一的兩個配置類註釋掉,在application.yml的最後面添加下面配置

cloud-provider:
  ribbon:
    NFLoadBalancerRuleClassName: com.example.cloudinvoker.MyRule
    NFLoadBalancerPingClassName: com.example.cloudinvoker.MyPing
    listOfServers: http://localhost:8080/,http://localhost:8081/

五、添加控制器 InvokerController.java

package com.example.cloudinvoker;

import org.springframework.cloud.client.loadbalancer.LoadBalanced;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.MediaType;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.client.RestTemplate;

@RestController
@Configuration
public class InvokerController {

    @LoadBalanced
    @Bean
    public RestTemplate getRestTemplate(){
        return new RestTemplate();
    }

    @RequestMapping(value="/router", method = RequestMethod.GET,produces = MediaType.APPLICATION_JSON_VALUE)
    public String router(){
        RestTemplate restTemplate = getRestTemplate();
        //根據名稱調用服務
        String json = restTemplate.getForObject("http://cloud-provider/", String.class);
        return json;
    }
}

六、測試

(1)啓動服務器端。
(2)啓動兩個服務提供者,在控制檯中分別輸入8080和8081啓動。
(3)啓動服務調用者。
(4)瀏覽器訪問http://localhost:9000/router,屢次刷新頁面,結果都是:

http://localhost:8081/

服務調用者項目IDEA控制檯定時輸出:

自定義服務器規則類,輸出服務器信息:
 localhost:8081
 localhost:8080
自定義Ping類,服務器信息:localhost:8081,狀態:true
自定義Ping類,服務器信息:localhost:8080,狀態:true
相關文章
相關標籤/搜索