一、什么是负载均衡?
负载均衡就是将一个操作单元(服务器、组件等)无法处理的工作负荷(访问请求,工作任务等),通过一个第三方负载分发组件,采用一定的分发策略,均匀的分配给多个工作单元执行。
二、手动实现负载均衡
2.1 创建服务提供者
创建两个服务提供者工程,要求:注册到Nacos中的服务名相同
配置文件 application.yml
:
server:
port: 9091
spring:
cloud:
nacos:
discovery:
server-addr: 192.168.52.131:8848 #nacos服务的地址
application:
name: ribbon-provider #向注册中心注册的名字
server:
port: 9092
spring:
cloud:
nacos:
discovery:
server-addr: 192.168.52.131:8848 #nacos服务的地址
application:
name: ribbon-provider #向注册中心注册的名字
2.2 创建服务消费者
创建工程:
配置文件 application.yml
:
server:
port: 8081
spring:
cloud:
nacos:
discovery:
server-addr: 192.168.52.131:8848 #nacos服务的地址
application:
name: ribbon-consumer #向注册中心注册的名字
Controller:
@RestController
@RequestMapping("/consumer")
public class ConsumerController {
//访问Rest服务的客户端
@Autowired
private RestTemplate restTemplate;
@Autowired
private DiscoveryClient discoveryClient;
private int currentIndex;
@RequestMapping(value="/getUserById/{id}")
public User getUserById(@PathVariable Integer id){
//根据服务名称从nacos中获取服务列表
List<ServiceInstance> serviceList = discoveryClient.getInstances("ribbon-provider");
//通过随机策略从服务列表中选出一个服务
//currentIndex = new Random().nextInt(serviceList.size());
//通过轮询策略从服务列表中选出一个服务
currentIndex = (currentIndex + 1) % serviceList.size();
ServiceInstance service = serviceList.get(currentIndex);
//从服务中获取服务的访问IP和端口号
String serviceUrl = service.getHost() + ":" + service.getPort();
//调用服务
String url = "http://"+serviceUrl+"/provider/getUserById/"+id;
return restTemplate.getForObject(url, User.class);
}
}
2.3 测试
启动服务:
向服务消费者发送请求,观察服务消费者调用的服务提供者:
三、Ribbon
3.1 Ribbon简介
- Spring Cloud Ribbon是基于Netflix Ribbon实现的一套客户端负载均衡工具;
- Nacos依赖中已经集成了Ribbon依赖,我们不需要额外引入Ribbon依赖;
3.2 基于Ribbon实现负载均衡
3.2.1 修改服务消费者工程ribbon_consumer
修改配置类ConfigBean
:
@Configuration
public class ConfigBean {
@Bean
/**
* Ribbon原理:添加了@LoadBalanced注解之后,Ribbon会给RestTemplate请求添加一个拦截器,
* 拦截器会获取请求中的ServiceId(服务名),然后根据服务名从注册中心获取服务列表,根据负载均衡策略
* 从服务列表中选中一个服务。再将请求中的服务名替换成选中服务的IP和端口号(ip:port),放行请求。
*/
@LoadBalanced //开启负载均衡,默认采取轮询策略
public RestTemplate restTemplate(){
return new RestTemplate();
}
//Ribbon默认采取轮询策略,若要替换,手动在此处注入相应策略对象
@Bean
public IRule iRule() {
return new RandomRule(); //随机策略
}
}
修改Controller:
@RestController
@RequestMapping("/consumer")
public class ConsumerController {
//访问Rest服务的客户端
@Autowired
private RestTemplate restTemplate;
@RequestMapping(value="/getUserById/{id}")
public User getUserById(@PathVariable Integer id){
// 获取服务名称
String serviceUrl = "ribbon-provider";
//调用服务
String url = "http://"+serviceUrl+"/provider/getUserById/"+id;
return restTemplate.getForObject(url, User.class);
}
}