Ribbon详解与实例

        Ribbon是一个为客户端提供负载均衡功能的服务,它内部提供了一个叫做ILoadBalance的接口代表负载均衡器的操作,比如有添加服务器操作、选择服务器操作、获取所有的服务器列表、获取可用的服务器列表等等。

需要解决的问题:

  • ① 如何在配置Eureka Client注册中心时不去硬编码Eureka Server的地址?
  • ② 在微服务不同模块间进行通信时,如何不去硬编码服务提供者的地址?
  • ③ 当部署多个相同微服务时,如何实现请求时的负载均衡? 

Ribbon是什么?

        Ribbon是Netflix发布的云中间层服务开源项目,其主要功能是提供客户端实现负载均衡算法。Ribbon客户端组件提供一系列完善的配置项如连接超时,重试等。简单的说,Ribbon是一个客户端负载均衡器,我们可以在配置文件中Load Balancer后面的所有机器,Ribbon会自动的帮助你基于某种规则(如简单轮询,随机连接等)去连接这些机器,我们也很容易使用Ribbon实现自定义的负载均衡算法。

下图展示了Eureka使用Ribbon时的大致架构:

SpringCloud之Ribbon入门案例

① 首先引入Ribbon依赖,Ribbon的使用依赖Eureka:

   <dependency>
     <groupId>org.springframework.cloud</groupId>
     <artifactId>spring-cloud-starter-eureka</artifactId>
   </dependency>
   <dependency>
     <groupId>org.springframework.cloud</groupId>
     <artifactId>spring-cloud-starter-ribbon</artifactId>
   </dependency>

② 如何使用Ribbon    

       使用RestTemplate进行Eureka Client(包括服务提供者以及服务消费者,在这里其实是服务消费者使用RestTemplate)之间的通信,为RestTemplate配置类添加@LoadBalanced注解即可,如下所示: 

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

主程序:

@SpringBootApplication
@EnableEurekaClient
//在启动该微服务的时候就能去加载我们的自定义Ribbon配置类,从而使配置生效
@RibbonClient(name="MICROSERVICECLOUD-DEPT")
public class DeptConsumer80_App{
	public static void main(String[] args){
		SpringApplication.run(DeptConsumer80_App.class, args);
	}
}

③ 如何解决硬编码

       使用添加@LoadBalanced注解后的RestTemplate调用服务提供者的接口时,可以使用虚拟IP替代真实IP地址。所谓的虚拟IP就是服务提供者在application.properties或yml文件中配置的spring.application.name属性的值。示例如下:

以前:

@RestController
public class DeptController_Consumer
{
	private static final String REST_URL_PREFIX = "http://localhost:8001"; //需要ip+端口

	@Autowired
	private RestTemplate restTemplate;

	@RequestMapping(value = "/consumer/dept/add")
	public boolean add(Dept dept)
	{
		return restTemplate.postForObject(REST_URL_PREFIX + "/dept/add", dept, Boolean.class);
	}
}

使用Ribbon后:

@RestController
public class DeptController_Consumer
{
	private static final String REST_URL_PREFIX = "http://MICROSERVICECLOUD-DEPT"; //微服务的虚拟id

	@Autowired
	private RestTemplate restTemplate;

	@RequestMapping(value = "/consumer/dept/add")
	public boolean add(Dept dept)
	{
		return restTemplate.postForObject(REST_URL_PREFIX + "/dept/add", dept, Boolean.class);
	}
}

小总结:Ribbon和Eureka整合后Consumer可以直接调用服务而不用再关心ip地址和端口号

微服务(服务提供者)集群搭建:

机器1
server:
  port: 8001

spring:
   application:
    name: microservicecloud-dept 
   datasource:
    type: com.alibaba.druid.pool.DruidDataSource          
    driver-class-name: org.gjt.mm.mysql.Driver            
    url: jdbc:mysql://localhost:3306/cloudDB01            
    username: root
    password: 123456

机器2
server:
  port: 8002

spring:
   application:
    name: microservicecloud-dept 
   datasource:
    type: com.alibaba.druid.pool.DruidDataSource   
    driver-class-name: org.gjt.mm.mysql.Driver          
    url: jdbc:mysql://localhost:3306/cloudDB02        
    username: root
    password: 123456

机器3
server:
  port: 8003

spring:
   application:
    name: microservicecloud-dept 
   datasource:
    type: com.alibaba.druid.pool.DruidDataSource         
    driver-class-name: org.gjt.mm.mysql.Driver            
    url: jdbc:mysql://localhost:3306/cloudDB03              
    username: root
    password: 123456

    其中{Spring.application.name}都是一样的,不可以变。

Ribbon组件IRule

默认的是RoundBobinRule(轮询)

RetryRule

  • 1、先按照RoundRobinRule(轮询)的策略获取服务,如果获取的服务失败侧在指定的时间会进行重试,进行获取可用的服务
  • 2、如多次获取某个服务失败,这不会再再次获取该服务如(高德地图上某条道路堵车,司机不会走那条道路)= 

使用:

@Configuration
public class ConfigBean //boot -->spring   applicationContext.xml --- @Configuration配置   ConfigBean = applicationContext.xml
{ 
	@Bean
	@LoadBalanced
	public RestTemplate getRestTemplate(){
		return new RestTemplate();
	}
	
	@Bean
	public IRule myRule(){
		//return new RoundRobinRule();
		//return new RandomRule();//达到的目的,用我们重新选择的随机算法替代默认的轮询。
		return new RetryRule();  //在这里选择负载均衡算法
	}
}

自定义负载均衡算法:

       所谓的自定义Ribbon Client的主要作用就是使用自定义配置替代Ribbon默认的负载均衡策略,注意:自定义的Ribbon Client是有针对性的,一般一个自定义的Ribbon Client是对一个服务提供者(包括服务名相同的一系列副本)而言的。自定义了一个Ribbon Client 它所设定的负载均衡策略只对某一特定服务名的服务提供者有效,但不能影响服务消费者与别的服务提供者通信所使用的策略。根据官方文档的意思,推荐在 springboot 主程序扫描的包范围之外进行自定义配置类。其实纯代码自定义RibbonClient的话有两种方式:

方式一:在springboot主程序扫描的包外定义配置类,然后为springboot主程序添加 @RibbonClient 注解引入配置类

配置类不应该在SpringBoot的包路径下通过@RibbonClient 注解加载:

@Configuration
public class MySelfRule
{
	@Bean
	public IRule myRule()
	{	
		return new RandomRule_ZY();  // 我自定义为每台机器5次,5次之后在轮询到下一个
	}
}

springboot主程序:

@SpringBootApplication
@EnableEurekaClient
//在启动该微服务的时候就能去加载我们的自定义Ribbon配置类,从而使配置生效
@RibbonClient(name="MICROSERVICECLOUD-DEPT",configuration=MySelfRule.class)
public class DeptConsumer80_App
{
	public static void main(String[] args)
	{
		SpringApplication.run(DeptConsumer80_App.class, args);
	}
}

自定义LoadBalance:

public class RandomRule_ZY extends AbstractLoadBalancerRule{

	// total = 0 // 当total==5以后,我们指针才能往下走,
	// index = 0 // 当前对外提供服务的服务器地址,
	// total需要重新置为零,但是已经达到过一个5次,我们的index = 1
	// 分析:我们5次,但是微服务只有8001 8002 8003 三台,OK?
	
	private int total = 0; 	    // 总共被调用的次数,目前要求每台被调用5次
	private int currentIndex = 0;	    // 当前提供服务的机器号

	public Server choose(ILoadBalancer lb, Object key){
		if (lb == null) {
			return null;
		}
		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;
			}
		
                        if(total < 5){
	                    server = upList.get(currentIndex);
	                    total++;
                        }else {
	                    total = 0;
	                    currentIndex++;
	                    if(currentIndex >= upList.size()){
	                      currentIndex = 0;
	                    }
                        }							
			if (server == null) {
				Thread.yield();
				continue;
			}

			if (server.isAlive()) {
				return (server);
			}

			// Shouldn't actually happen.. but must be transient or a bug.
			server = null;
			Thread.yield();
		}
		return server;
	}
	@Override
	public Server choose(Object key){
		return choose(getLoadBalancer(), key);
	}

	@Override
	public void initWithNiwsConfig(IClientConfig clientConfig){
	}
}

 

 

 

  • 43
    点赞
  • 266
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
Ribbon组合框是一种常见的图形用户界面控件,常用于显示和选择多个选项中的一个。它通常与Ribbon界面风格相结合,使用户能够在菜单和工具栏之外获得更多的操作选项。 Ribbon组合框具有以下几个主要特点。首先,它具有下拉列表的功能,用户可以通过点击或下拉箭头来打开下拉菜单,从中选择一个选项。其次,它支持文本和图标显示,可以根据需要显示选项的文本描述或使用图标来进行直观展示。再次,它可以自动完成用户的输入,即根据用户键入的信息来筛选和匹配可选项,提供更加智能的交互体验。 Ribbon组合框的应用场景非常广泛。例如,在表单设计中,可以使用Ribbon组合框来选择性别、部门、职位等信息;在电商网站中,可以使用Ribbon组合框来选择商品的分类、品牌或价格范围;在音乐播放器中,可以使用Ribbon组合框来选择歌曲的艺术家、专辑或流派。总之,无论是选择固定选项,还是根据输入内容进行筛选,Ribbon组合框都是一种非常便捷和直观的控件。 在使用Ribbon组合框时,需要注意以下几点。首先,要保持选项的清晰和有序,避免选项数量过多或杂乱无章;其次,要根据界面整体布局和风格设计,选择合适的颜色、尺寸和样式;再次,要为组合框提供适当的提示和帮助信息,如下拉菜单的简洁说明或自动完成的操作提示。 总之,Ribbon组合框是一种强大的用户界面控件,提供了丰富的选项选择和交互功能,在各种应用场景下都能发挥其作用。通过合理使用和设计,可以营造出更加方便和高效的用户体验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值