一、Ribbon简介
1、Ribbon介绍
Ribbon是由Netflix公司推出的开源软件,是基于HTTP和TCP协议的,其主要功能是实现客户端软件的负载均衡算法。
Spring Cloud中Ribbon就是基于Netflix公司的Ribbon实现的。它不需要单独部署,但是却存在于整个微服务中。Eureka里面有Ribbon,OpenFeign也是基于Ribbon实现的。
2、Ribbon原理
内部基于ILoadBalancer实现的(代码层面)。
继承关系如下:
使用Ribbon工作原理:
所有的项目都会注册到Eureka中,Eureka允许不同项目的spring.application.name是相同。当相同时会认为这些项目一个集群。所以同一个项目部署多次时都是设置应用程序名相同。
Application Client会从Eureka中根据spring.application.name加载Application Service的列表。根据设定的负载均衡算法,从列表中取出一个URL,到此Ribbon的事情结束了。剩下的事情由程序员自己进行技术选型,选择一个HTTP协议工具,通过这个URL调用Application Service。
注意:以下事情和Ribbon没有关系的
- Application Service注册到Eureka过程。这是Eureka的功能。
- Application Client从Eureka取出注册列表。这是Eureka的功能。
- Application Client 通过URL访问Application Service。具体实现可以自己进行选择使用哪个HTTP工具。
只有Application Client从Eureka中取出列表后进行负载均衡算法的过程和Ribbon有关系。
二、负载均衡解决方案分类及特征
业界主流的负载均衡解决方案有:集中式负载均衡和线程内负载均衡。
1.1 集中式负载均衡
即在客户端和服务端之间使用独立的负载均衡设施(可以是硬件,如F5, 也可以是软件,如nginx), 由该设施负责把访问请求通过某种策略转发至服务端。
也叫做:服务器端负载均衡。1.2 进程内负载均衡
将负载均衡逻辑集成到客户端组件中,客户端组件从服务注册中心获知有哪些地址可用,然后自己再从这些地址中选择出一个合适的服务端发起请求。Ribbon就是一个进程内的负载均衡实现。
也叫做:客户端负载均衡。三、微服务架构中Ribbon
在Application Client中配置Ribbon。
微服务架构既然把项目拆分成多个项目,一定会出现项目调用另外一个项目的情况。
在加上Eureka支持多项目同项目名(spring.application.name)所以这个时候就可以使用Ribbon。
项目B和项目C实际上是一个项目。项目A调用项目B和项目C,那么Ribbon应该配置在项目A中(Application Client中)
四、搭建Application Service集群
前提:已经配置了单机版Eureka。端口为8761
新建项目ApplicationServiceDemo1、添加依赖
新建项目后,在pom.xml中添加依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
2、编写配置文件
在resources下新建application.yml.
spring:
application:
name: applicationservice
eureka:
client:
service-url:
defaultZone: http://localhost:8888/eureka/
server:
port: 8081
3、编写控制器
@RestController
public class MyController {
@RequestMapping("/demo01")
public String demo01(){
return "bjsxt111";
}
@RequestMapping("/demo02")
public String demo02(String name,String pwd){
System.out.println(name+"--"+pwd);
return "bjsxt";
}
@RequestMapping("/demo03")
public String demo03(@RequestBody User user){
System.out.println(user);
return "bjsxt";
}
@RequestMapping("/demo04/{name}/{pwd}")
public String demo04(@PathVariable String name,@PathVariable String pwd){
System.out.println(name+"--"+pwd);
return "bjsxt";
}
@PostMapping("/selectAll")
public List<User> selectAll(){
List<User> list=new ArrayList<>();
list.add(new User("zs","123"));
list.add(new User("lisi","123"));
list.add(new User("sxt","123"));
return list;
}
@RequestMapping("/selectMore")
public List<String> selectMore(@RequestBody List<Integer> id){
List<String> list = new ArrayList<>();
for (Integer s : id) {
list.add("sxt:"+s);
}
return list;
}
@RequestMapping("/selectA")
public List<String> selectA(@RequestBody List<String> list){
System.out.println(list);
List<String> list1=new ArrayList<>();
for (String s:list) {
list1.add(s+"查询数据");
}
return list1;
}
}
4、编写启动类
@SpringBootApplication
@EnableEurekaClient
public class EurekaApplicationserverApplication {
public static void main(String[] args) {
SpringApplication.run(EurekaApplicationserverApplication.class, args);
}
}
5、启动项目(启动三个)
修改启动配置文件
添加 -Dserver.port=8082
添加 -Dserver.port=8083
观察Eureka管理页面会发现已经注册了三个Provider
五、基于Ribbon测试负载均衡
1、添加依赖
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
2、编写配置文件
eureka:
client:
service-url:
defaultZone: http://localhost:8888/eureka/
server:
port: 9999
spring:
application:
name: application-client
3、编写service
新建com.bjsxt.service.ClientService及实现类。
public interface ClientService {
/**@return 返回值类型不是必须和调用的控制器方法返回值一样,需要看页面要什么。*/
String client();
}
@Service
public class ClientServiceImpl implements ClientService {
@Autowired
private LoadBalancerClient loadBalancerClient;
@Override
public String client() {
ServiceInstance applicationservice = loadBalancerClient.choose("applicationservice");
//获取URL地址
System.out.println(applicationservice.getUri());
/* // 获取Application Service IP
System.out.println(applicationservice.getHost());
// 获取Ip及端口。
System.out.println(applicationservice.getInstanceId());
// 获取额外声明信息.{management.port=xxxx}
System.out.println(applicationservice.getMetadata());
// 端口 8082
System.out.println(applicationservice.getPort());
// 模式 null
System.out.println(applicationservice.getScheme());
// 应用程序名 applicationservice
System.out.println(applicationservice.getServiceId());*/
return null;
}
}
4、编写控制器
新建控制器com.bjsxt.controller.ClientController
@RestController
public class ClientController {
@Autowired
private ClientService clientService;
@RequestMapping("/client")
public String client(){
clientService.client();
return "aa";
}
}
5、编写启动类
新建com.bjsxt.ClientApplication。
@SpringBootApplication
@EnableEurekaClient
public class EurekaApplicationclientApplication {
public static void main(String[] args) {
SpringApplication.run(EurekaApplicationclientApplication.class, args);
}
}
6、运行
运行程序,通过浏览器多次访问项目ApplicationClientDemo中/client控制器,观察IDEA控制器中输出结果。
会发现访问是轮询访问的。
六、Ribbon支持的负载均衡算法
ribbon的负载均衡策略是通过不同的类型来实现的,下表详细介绍一些常用负载均衡策略及对应的Ribbon策略类。
id | 策略名称 | 策略对应的类名 | 实现原理 |
---|---|---|---|
1 | 轮询策略(默认) | RoundRobinRule | 轮询策略表示每次都按照顺序取下一个application service,比如一共有5个application service,第1次取第1个,第2次取第2个,第3次取第3个,以此类推 |
2 | 权重轮询策略(常用,中小型项目使用) | WeightedResponseTimeRule | 1.根据每个application service的响应时间分配一个权重,响应时间越长,权重越小,被选中的可能性越低。2.原理:一开始为轮询策略,并开启一个计时器,每30秒收集一次每个application service的平均响应时间,当信息足够时,给每个application service附上一个权重,并按权重随机选择application service,权重越高的application service会被高概率选中。 |
3 | 随机策略(不推荐,测试使用,开发一般不使用) | RandomRule | 从application service列表中随机选择一个 |
4 | 最少并发数策略(应用在硬件软件环境一致的情况下,中小型项目使用) | BestAvailableRule | 选择正在请求中的并发数最小的application service,除非这个application service在熔断中。 |
5 | 重试策略。在“选定的负载均衡策略”基础上进行重试机制 | RetryRule | 1.“选定的负载均衡策略”这个策略是轮询策略RoundRobinRule 2.该重试策略先设定一个阈值时间段,如果在这个阈值时间段内当选择application service不成功,则一直尝试采用“选定的负载均衡策略:轮询策略”最后选择一个可用的application service |
6 | 可用性敏感策略(一般在同区域内服务集群环境中使用) | AvailabilityFilteringRule | 过滤性能差的application service,有2种: 第一种:过滤掉在eureka中处于一直连接失败application service第二种:过滤掉高并发的application service |
7 | 区域敏感性策略(应用在大型的,物理隔离分布式环境中) | ZoneAvoidanceRule | 1.以一个区域为单位考察可用性,对于不可用的区域整个丢弃,从剩下区域中选可用的application service 2.如果这个ip区域内有一个或多个实例不可达或响应变慢,都会降低该ip区域内其他ip被选中的权重。 |
七、 指定负载均衡策略
1、添加bean
在application client的配置类中添加。配置类中指定哪个负载均衡策略默认使用哪种策略。不允许配置多个负载均衡策略的实例。
com.bjsxt.config.ApplicationConfig
@Configuration
public class ApplicationConfig {
@Bean
public RandomRule getRandomRule(){
return new RandomRule();
}
}
2、在ApplicationService观察输出语句
八、RestTemplate
RestTemplate是spring-web-xxx.jar包中提供的Http协议实现类。也就是说导入spring-boot-starter-web的项目可以直接使用RestTemplate类,就是基于模板方法设计模式的,封装了所有需要使用的API
在该类中主要针对6类请求方式封装的方法。
1、说明
get方式提供了两个方法:
两个方法都是发送get请求并处理响应。区别:
getForObject:把响应体直接转换为对象。该方法返回值为特定类类型。舍弃了Response Header的东西,但是用起来比getForEntity方便。如果只需要获取响应体中内容(调用控制器方法的返回值)使用此方法。
getForEntity:返回值包含响应头和响应体。用起来比getForObject稍微麻烦一些。
注意:
如果方法返回值是String或基本数据类型时,建议给定produces设置响应结果类型,否则使用浏览器测试和使用RestTemplate获取的ContentType类型可能不一致。
2、get方式
@SpringBootTest
class EurekaApplicationclientApplicationTests {
//get/post/通用
@Test
void contextLoads() {
RestTemplate rp=new RestTemplate();
//发送get请求
//rp.getForEntity() rp.getForObject()这两个方法都可以发送get请求
//rp.getForEntity()返回结果中包含了响应头响应体等信息
//rp.getForObject()返回结果中只包含了响应的结果
System.out.println("------getForEntity()和rp.getForObject()区别-------------");
//String forObject = rp.getForObject("http://192.168.237.1:8081/demo01", String.class);
//System.out.println(forObject);
//ResponseEntity<String> forEntity = rp.getForEntity("http://192.168.237.1:8081/demo01", String.class);
//System.out.println(forEntity.getBody());
//System.out.println(forEntity.getHeaders());
System.out.println("------------普通数据发送-----------------");
String name="zs";
String pwd="123";
//String forObject = rp.getForObject("http://192.168.237.1:8081/demo02?zs=" + name + "&pwd=" + pwd, String.class);
//String forObject = rp.getForObject("http://192.168.237.1:8081/demo02?zs={1}&pwd={2}", String.class,name,pwd);
System.out.println("------------对象数据发送 注意:get请求不支持对象,传递的时候需要把对象拆分-----------------");
User user=new User("lisi","123");
//rp.getForObject("http://192.168.237.1:8081/demo03?name={1}&pwd={2}",String.class,user.getName(),user.getPwd());
System.out.println("------------Rest风格方式-----------------");
//rp.getForObject("http://192.168.237.1:8081/demo04/{1}/{2}",String.class,user.getName(),user.getPwd());
System.out.println("------------Map类型数据传输-----------------");
Map<String,String> map =new HashMap<>();
map.put("a","value-a");
map.put("b","value-b");
rp.getForObject("http://192.168.237.1:8081/demo04/{a}/{b}",String.class,map);
}
}
3、post方式
@SpringBootTest
class EurekaApplicationclientApplicationTests {
//发送Post请求
@Test
void demoPost(){
RestTemplate rp=new RestTemplate();
//相比get而言post中多了一个rp.postForLocation()方法 这个方法返回值是一个URL对象 这个对象处理的时候比较麻烦 几乎不用
System.out.println("---------普通类型数据传递-------------");
String name="zs";
String pwd="123";
//rp.postForObject("http://192.168.237.1:8081/demo02?zs={1}&pwd={2}",null,String.class,name,pwd);
System.out.println("------------post传递请求体数据也就是可以直接传递对象---------------");
//发送是请求体数据 再applicationservice 中进行数据接收时候必须使用@RequestBody才可以
//发送请求的时候即可以传递普通数据又可以传递请求体数据
User user=new User("lisi","123");
rp.postForObject("http://192.168.237.1:8081/demo03",user,String.class);
}
}
4、exchange
当请求的控制器返回值类型为List
除此以外,如果需要设置请求头参数情况也需要使用exchange方法。
@SpringBootTest
class EurekaApplicationclientApplicationTests {
//【A】其他的请求如果发送
//【B】接收集合类型返回值 数据没有办法遍历
@Test
void getExchange(){
RestTemplate rp=new RestTemplate();
//List list = rp.getForObject("http://192.168.237.1:8081/selectAll", List.class);
//System.out.println(((User)list.get(0)).getName());
//设置请求体数据
User user=new User("zs","123");
HttpEntity<User> httpEntity = new HttpEntity<>(user);
//设置响应数据类型
ParameterizedTypeReference<List<User>> pt=new ParameterizedTypeReference<List<User>>(){};
ResponseEntity<List<User>> exchange = rp.exchange("http://192.168.237.1:8081/selectAll", HttpMethod.GET, httpEntity, pt);
//返回响应体数据 就是我们需要的类型
List<User> list = exchange.getBody();
for (User u:list) {
System.out.println(u);
}
}
}
九、实现Application Client调用Application Service
上面的例子中是LoadBalance调用Provider的测试。现在实现调用Provider的控制器接口方法。
在Java代码中调用另一个控制器接口,可以使用之前学习的HttpClient实现,在今天的课程中换一种技术实现,基于RestTemplate完成的。
1、新建配置类
新建com.bjsxt.config.RibbonConfig。
注意方法上面要有@LoadBalanced注解。否则Ribbon不生效。
@Configuration
public class RibbonConfig {
@Bean
public RandomRule getRandomRule(){
return new RandomRule();
}
@Bean
@LoadBalanced
public RestTemplate getRestTemplate(){
return new RestTemplate();
}
}
2、修改service实现类
注意:
无论使用RestTemplate的哪个方法,方法中URL参数必须使用spring.application.name应用程序名(ServerID)替换原来URL中host(主机)和port(端口)。因为底层Ribbon是根据应用程序名获取注册列表的。
@Service
public class ClientServiceImpl implements ClientService {
@Autowired
private RestTemplate restTemplate;
@Override
public String client() {
String forObject = restTemplate.getForObject("http://applicationservice/demo01", String.class);
System.out.println("结果为:"+forObject);
return null;
}
3、运行
controller代码
@RestController
public class ClientController {
@Autowired
private ClientService clientService;
@RequestMapping("/client")
public String client(){
clientService.client();
return "aa";
}
}
运行项目,访问/client控制器,观察IDEA控制台打印的内容。