SpringCloud第三章:客户端负载均衡与服务调用

正常本章节应该讲解Netflix Ribbon,由于从Netflix Eureka Client 3.0版本开始内置Ribbon实现机制,不需要单独依赖Ribbon,如果引入Ribbon会报错 java.lang.IllegalStateException: No instances available for XXXXXX,并且想要修改或自定义负载策略也没有IRule接口可以实现,后面我会讲解到这些内容。

学习内容

  • 什么是负载均衡
  • Netflix Eureka Client 怎么实现服务调用
  • Netflix Eureka Client 怎么实现负载均衡
  • 怎么修改负载均衡策略
  • 怎么自定义负载均衡策略

负载均衡

在任何一个系统中,负载均衡都是一个十分重要且不得不去实施的内容,它是系统处理高并发、缓解网络压力和服务端扩容的重要手段之一。
负载均衡(Load Balance) ,简单点说就是将用户的请求平摊分配到多个服务器上运行,以达到扩展服务器带宽、增强数据处理能力、增加吞吐量、提高网络的可用性和灵活性的目的。
常用负载均衡有两种:

  1. 服务端负载均衡
  2. 客户端负载均衡

服务端负载均衡

服务端负载均衡在微服务没有流行起来之前是最常见的负载均衡方式,其工作原理如下图。


Nginx负载均衡

服务端负载均衡是在客户端和服务端之间建立一个独立的负载均衡服务器,该服务器既可以是硬件设备(例如 F5),也可以是软件(例如 Nginx)。这个负载均衡服务器维护了一份可用服务端清单,然后通过心跳机制来删除故障的服务端节点,以保证清单中的所有服务节点都是可以正常访问的。
当客户端发送请求时,该请求不会直接发送到服务端进行处理,而是全部交给负载均衡服务器,由负载均衡服务器按照某种算法(例如轮询、随机等),从其维护的可用服务清单中选择一个服务端,然后进行转发。

服务端负载均衡具有以下特点:

  • 需要建立一个独立的负载均衡服务器。
  • 负载均衡是在客户端发送请求后进行的,因此客户端并不知道到底是哪个服务端提供的服务。
  • 可用服务端清单存储在负载均衡服务器上。

客户端负载均衡

客户端负载均衡是随着微服务发展而流行起来的,工作原理如下图:


客户端负载均衡

客户端负载均衡是将负载均衡逻辑以代码的形式封装到客户端上,即负载均衡器位于客户端。客户端通过服务注册中心(例如 Eureka Server)获取到一份服务端提供的可用服务清单。有了服务清单后,负载均衡器会在客户端发送请求前通过负载均衡算法选择一个服务端实例再进行访问,以达到负载均衡的目的;
客户端负载均衡也需要心跳机制去维护服务端清单的有效性,这个过程需要配合服务注册中心一起完成。

客户端负载均衡具有以下特点:

  • 负载均衡器位于客户端,不需要单独搭建一个负载均衡服务器。
  • 负载均衡是在客户端发送请求前进行的,因此客户端清楚地知道是哪个服务端提供的服务。
  • 客户端都维护了一份可用服务清单,而这份清单都是从服务注册中心获取的。

Eureka Client(Ribbon) 就是一个基于 HTTP 和 TCP 的客户端负载均衡器,Eureka Client 会从 Eureka Server(服务注册中心)中获取服务端列表,然后通过负载均衡策略将请求分摊给多个服务提供者,从而达到负载均衡的目的。

Netflix Eureka Client 实现服务调用

Eureka Client与 RestTemplate(Rest 模板)配合使用,以实现微服务之间的调用。

RestTemplate 是 Spring 家族中的一个用于消费第三方 REST 服务的请求框架。RestTemplate 实现了对 HTTP 请求的封装,提供了一套模板化的服务调用方法。通过它,Spring 应用可以很方便地对各种类型的 HTTP 请求进行访问。

RestTemplate 针对各种类型的 HTTP 请求都提供了相应的方法进行处理,例如 HEAD、GET、POST、PUT、DELETE 等类型的 HTTP 请求,分别对应 RestTemplate 中的 headForHeaders()、getForObject()、postForObject()、put() 以及 delete() 方法。

废话不多说我们上代码

  1. 在主工程 SpringCloud 下,创建一个名为 springcloud-consumer-user-80 的微服务,并在其 pom.xml 中引入所需的依赖
<?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 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <artifactId>SpringCloud</artifactId>
        <groupId>com.heitaokei</groupId>
        <version>1.0-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>
    <artifactId>springcloud-consumer-user-80</artifactId>
    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <!--引入 Eureka Client 的依赖,将服务注册到 Eureka Server-->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
        </dependency>
        <!-- Spring Boot 监控模块-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-actuator</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-devtools</artifactId>
            <scope>runtime</scope>
            <optional>true</optional>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>
        <dependency>
            <artifactId>springcloud-api</artifactId>
            <groupId>com.heitaokei</groupId>
            <version>${project.version}</version>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>
</project>
  1. 在resource下,创建配置文件 application.yml,配置内容如下
server:
  port: 80
eureka:
  client:
    register-with-eureka: false #本微服务为服务消费者,不需要将自己注册到服务注册中心
    fetch-registry: true  #本微服务为服务消费者,需要到服务注册中心搜索服务
    service-url:
      defaultZone: http://localhost:7001/eureka/
  1. 在 com.heitaokei 包下,创建一个名为 SpringCloudConsumer_80的启动类,并使用 @EnableEurekaClient开启Eureka客户端功能
package com.heitaokei;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;

@SpringBootApplication
@EnableEurekaClient
public class SpringCloudConsumer_80 {
    public static void main(String[] args) {
        SpringApplication.run(SpringCloudConsumer_80.class, args);
    }
}
  1. 在 com.heitaokei.config 包下,创建一个名为 ConfigBean 的配置类,将 RestTemplate 注入到容器中
package com.heitaokei.config;

import com.heitaokei.MyLoadBalancerConfiguration;
import org.springframework.cloud.client.loadbalancer.LoadBalanced;
import org.springframework.cloud.loadbalancer.annotation.LoadBalancerClient;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.client.RestTemplate;

@Configuration
public class ConfigBean {
    @Bean
    @LoadBalanced // 开启负载均衡
    public RestTemplate getRestTemplate() {
        return new RestTemplate();
    }
}
  1. 在 com.heitaokei.controller 包下,创建一个名为 UserConsumerController 的 Controller,该 Controller 中定义的请求用于调用服务端提供的服务
package com.heitaokei.controller;

import com.heitaokei.entity.User;
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;

/**
 * 用户消费者
 */
@RestController
public class UserConsumerController {
    //面向微服务编程,即通过微服务的名称来获取调用地址 使用注册到 Spring Cloud Eureka 服务注册中心中的服务,*即服务提供者中 application.name*
    private static final String PROVIDER_URL_PREFIX = "http://PROVIDER-USER";
    @Autowired
    private RestTemplate restTemplate; // 注册ConfigBean配置的负载均衡bean

    @GetMapping(value = "/user")
    public User getUser() {
        return restTemplate.getForObject(PROVIDER_URL_PREFIX + "/user", User.class);
    }
}

由于要测试客户端负载均衡所以把springcloud-provider-user-8001项目中配置文件application.name修改成provider-user,具体如下图所示


服务提供者的应用名

Eureka注册中心 服务器提供者应用名
  1. 依次启动服务注册中心 springcloud-eureka-7001、服务提供者 springcloud-provider-user-8001 以及服务消费者 springcloud-consumer-user-80
  2. 使用浏览器访问 http://localhost/user,结果如下图,能够正常返回结果表示服务调用成功
    服务调用接口

Netflix Eureka Client 实现负载均衡

Eureka Client 内置了客户端的负载均衡器,能够轻松地实现客户端的负载均衡。Eureka Client会先从 Eureka Server(服务注册中心)去获取服务端列表,然后通过负载均衡策略将请求分摊给多个服务端,从而达到负载均衡的目的
Ribbon 提供了一个 IRule 接口,但是Eureka Client是没有这个接口的,Eureka Client负载策略接口为ReactorServiceInstanceLoadBalancer,它只内置两种负载策略,轮询策略(默认)、随机策略


内置两种策略

下面通过实例介绍一下客户端负载的应用

  1. 参考 springcloud-provider-user-8001,再创建两个微服务 Moudle :springcloud-provider-user-8002 和 springcloud-provider-user-8003
  2. 在 springcloud-provider-user-8002 中 application.yml 中,修改端口号
server:
  port: 8002
spring:
  application:
    name: provider-user #对外暴漏的微服务名称 不做修改,与springcloud-provider-user-8001一致
  1. 在 springcloud-provider-user-8002 中修改UserController中getUser返回值,方便后面区分调用的是哪个服务
package com.springcloud.controller;
import com.heitaokei.entity.User;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class UserController {
    @GetMapping(value = "/user")
    public User getUser() {
        User user = new User();
        user.setUserName("用户222222");
        user.setAge(22);
        user.setAddress("火星");
        return user;
    }
}
  1. 在 springcloud-provider-user-8003 中 application.yml 中,修改端口号
server:
  port: 8003
spring:
  application:
    name: provider-user #对外暴漏的微服务名称 不做修改,与springcloud-provider-user-8001一致
  1. 在 springcloud-provider-user-8003 中修改UserController中getUser返回值,方便后面区分调用的是哪个服务
package com.springcloud.controller;
import com.heitaokei.entity.User;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class UserController {
    @GetMapping(value = "/user")
    public User getUser() {
        User user = new User();
        user.setUserName("用户333");
        user.setAge(33);
        user.setAddress("地球村");
        return user;
    }
}
  1. 依次启动 springcloud-eureka-7001/7002/7003(服务注册中心集群,也可只启动一个,我就只启动一个因为比较吃内存,如果只启动一个可以把Eureka注册地址改成一个即可)、springcloud-provider-user-8001/8002/8003(服务提供者集群)以及 springcloud-consumer-user-80(服务消费者)
    6.使用浏览器访问 http://localhost/user,返回结果后继续刷新该请求,会发现每次返回的结果是从不同的服务提供者返回的,记录一下返回结果即可发现返回结果出现的顺序一直是一致的,例如:3-1-2-3-1-2-3-1-2-3-1所以我们确定Eureka Client 默认负载均衡策略为轮询

怎么修改负载策略

如果是Ribbon使用内置策略只需要重新注入IRule即可,但是Eureka Client确不是这样的,Eureka Client需要在启动类同级创建一个配置类

  1. 我们把策略改为随机,在springcloud-consumer-user-80模块的com.heitaokei包下创建一个MyLoadBalancerConfiguration类
package com.heitaokei;

import org.springframework.cloud.client.ServiceInstance;
import org.springframework.cloud.loadbalancer.core.RandomLoadBalancer;
import org.springframework.cloud.loadbalancer.core.ReactorLoadBalancer;
import org.springframework.cloud.loadbalancer.core.ServiceInstanceListSupplier;
import org.springframework.cloud.loadbalancer.support.LoadBalancerClientFactory;
import org.springframework.context.annotation.Bean;
import org.springframework.core.env.Environment;

/**
 * 自定义负载配置
 */
public class MyLoadBalancerConfiguration {
    @Bean
    ReactorLoadBalancer<ServiceInstance> randomLoadBalancer(Environment environment,
                                                            LoadBalancerClientFactory loadBalancerClientFactory) {
        String name = environment.getProperty(LoadBalancerClientFactory.PROPERTY_NAME);
        // 随机
        return new RandomLoadBalancer(loadBalancerClientFactory.getLazyProvider(name, ServiceInstanceListSupplier
                .class), name);
    }
}
  1. 在启动类SpringCloudConsumer_80 加上注解 @LoadBalancerClients,代码如下
@SpringBootApplication
@EnableEurekaClient
@LoadBalancerClients(defaultConfiguration = MyLoadBalancerConfiguration.class)
public class SpringCloudConsumer_80 {
    public static void main(String[] args) {
        SpringApplication.run(SpringCloudConsumer_80.class, args);
    }
}
  1. 重新启动springcloud-consumer-user-80,浏览器访问 http://localhost/user,返回结果后继续刷新该请求,我们会发现返回结果是没有规律的,证明我们修改为随机的负载均衡策略生效了

怎么自定义负载均衡策略

自定义策略我们只需要自己创建一个ReactorServiceInstanceLoadBalancer接口的实现类,实现choose方法即可,具体实现的算法需要根据具体需求来确定,一般默认就够了,所以这里就不演示实现算法了,因为实现方式已经知道了,具体实现就不操作了

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容