1.Ribbon 是什么
- SpringCloud Ribbon 是基于 Netflix Ribbon 实现的一套客户端负载均衡的工具
- 主要功能是提供客户端的软件负载均衡算法,将 Netflix 的中间服务处连接在一起
- Ribbon的客户端组件提供一系列完整的配置项,如:连接超时、重试等。简单的说就是在配置文件中列出 LoadBalance
(LB:负载均衡)后面所有的机器,Ribbon 会自动帮你基于某种规则去连接这些机器
1.2 负载均衡(LB) - 核心作用:就是将用户的请求平摊在多个服务器上,从而达到系统的 High Availability(HA :高可用)
- 常见的负载均衡软件有 Nginx、Lvs 等
- Dubbo 和 SpringCloud 都提供了负载均衡,SpringCloud 的负载均衡算法可以自定义
- 负载均衡算法实现的方式:轮询、权重、随机
- 负载均衡的简单分类:
1、集中式
在服务的 消费者 和 提供者之间使用独立的 LB 设施,如Nginx(反向代理服务器) ,由该设施负载把访问请求通过某种策略发送至服务的提供者
2、进程式
将LB逻辑集成到 消费者,消费者 从服务注册中心得到哪些地址可用,然后自己再从这些地址中选出一个合适服务器
Ribbon 就属于进程内 LB ,它只是一个类库,集成与消费者进程,消费者通过他来获取到服务提供者的地址
2.1 整合 Ribbon
- 在 POM 文件中添加 Ribbon、Eureka 依赖
<!--ribbon--><!-- https://mvnrepository.com/artifact/org.springframework.cloud/spring-cloud-starter-ribbon --><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-ribbon</artifactId><version>1.4.7.RELEASE</version></dependency><!--Eureka 服务提供者--><!-- https://mvnrepository.com/artifact/org.springframework.cloud/spring-cloud-starter-eureka --><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-eureka</artifactId><version>1.4.7.RELEASE</version></dependency>
2.编写 application.yml
# Eurekaeureka:client:# 不向注册中心注册自己register-with-eureka: false# 配置 可连接的注册中心service-url:defaultZone: http://127.0.0.1:7001/eureka/,http://127.0.0.2:7002/eureka/
3.在主启动类上添加 Eureka 启动项 : @EnableEurekaClient
4.在之前把 RestTemplate 注册到 Bean 的配置方法上添加一个注解
5.修改控制层
/**提供者 URL 的前缀 * * 不使用 Ribbon 时 ,这里就是第一中写法 * * 使用 Ribbon 实现负载均衡 时,这里就不能写死为一个地址, * 而需要通过注册中心的服务名来访问 * 服务名:在 提供者 YML 文件中配置的 spring:application:name: 的值 *或者 监控页面的 Application 字段值 *///private static final String REST_URL_PREFIX = "http://localhost:8081";private static final String REST_URL_PREFIX = "http://SpringCloud-02-provider";
3.自定义策略(简单示例)
- 可以点开刚刚看的那个 RandomRule 的源代码,复制过来修改一下
- 修改要求:每个提供者访问五次
public class DiyRule extends AbstractLoadBalancerRule {public Server choose(ILoadBalancer lb, Object key) {if (lb == null) {return null;} else {Server server = null;while(server == null) {if (Thread.interrupted()) {return null;}// 获得可获得(活着的)的服务List<Server> upList = lb.getReachableServers();// 获得所有的服务List<Server> allList = lb.getAllServers();int serverCount = allList.size();if (serverCount == 0) {return null;}//==上面是写死的======中间是修改部分==================System.out.println("自定义的 Rule");System.out.println(upList.size());// 访问某一个提供者的次数int times = 0;// 提供者的下标int index = 0;// 从活着的服务中随机获取一个server = (Server)upList.get(index);if (times < 4){times++;}else {times = 1;index = (index + 1) % upList.size();}//==下面是写死的======中间是修改部分===================if (server == null) {Thread.yield();} else {if (server.isAlive()) {return server;}server = null;Thread.yield();}}return server;}}@Overridepublic void initWithNiwsConfig(IClientConfig iClientConfig) {}@Overridepublic Server choose(Object o) {return null;}}
2.MyRule 类
@Configurationpublic class CustomizedRule {/** * 修改默认的负载均衡策略 */@Beanpublic IRule customize(){// 先使用已经实现的策略——随机return new DiyRule();}}
3.在主启动类上添加
// configuration:标注 Rule 的配置类 ; name:标注需要配置的服务名@RibbonClient(name = "SPRINGCLOUD-PROVIDER", configuration = CustomizedRule.class)
Eureka 和 Ribbon 整合以后,在消费端,就不需要关系请求服务的 IP地址 和 端口号了,就只需要知道服务名称,直接调用即可