springcloud 服务间通讯方式 Ribbon

Easter79
• 阅读 217

查看Ribbon :https://blog.csdn.net/qq_32534855/article/details/84111188

1.Eureka服务发现

springcloud 服务间通讯方式 Ribbon

product 启动了两个实例

2.RestTemplate

RestTemplate

参考:https://blog.csdn.net/madmk/article/details/76431486

        RestTemplate restTemplate = new RestTemplate();
        String response =   restTemplate.getForObject("http://localhost:9001/msg",String.class);
        log.info("response={}",response);

这种方式,调用很简单,但是访问地址写死,不方便,所以这种方式很少用。

并且,当服务启动多个时,很难做到调用多个服务实例。

3.LoadBalancerClient + RestTemplate

我们通过loadBalancerClient根据应用名获取url 


    @Autowired
    LoadBalancerClient loadBalancerClient;

    /**
     * LoadBalancerClient + RestTemplate方式
     *
     * @return
     */
    @GetMapping("/msg2")
    public String helloMsg2() {
        RestTemplate restTemplate = new RestTemplate();

        ServiceInstance instance = loadBalancerClient.choose("product");
        String storesUri = String.format("http://%s:%s", instance.getHost(), instance.getPort());

        String response = restTemplate.getForObject(storesUri + "/msg", String.class);
        log.info("response={}", response);

        return response;
    }

4.@LoadBalanced注解 

添加RestTemplateConfig 配置文件,重点在方式上添加**@LoadBalanced**注解

@Configuration
public class RestTemplateConfig {

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

使用

    @Autowired
    private RestTemplate restTemplate;

    /**
     * LoadBalancerClient + RestTemplate方式
     *
     * @return
     */
    @GetMapping("/msg3")
    public String helloMsg3() {

        //通过应用名+访问地址
        String response = restTemplate.getForObject("http://product/msg", String.class);
        log.info("response={}", response);

        return response;
    }

5.Ribbon实现客户端负载均衡原理

5.1主要概念

  • 服务发现 :根据服务名字,把该服务下所以实力查询出来
  • 服务选择规则:根据服务选择规则,选择出一条有效服务
  • 服务监听:检测失效的服务,高效剔除

5.2主要组件

  • ServerList
  • IRule
  • ServerListFilter

流程:通过ServerList获取可用服务列表,然后通过ServerListFilter过滤掉一部分服务,最后通过IRule选择出一个最终目标结果。

5.3源码分析

5.3.1类间关系

springcloud 服务间通讯方式 Ribbon

5.3.2 获取可用服务列表

  ServiceInstance instance = loadBalancerClient.choose("product");

我们按住CTRL跟进到choose方法里面

    public ServiceInstance choose(String serviceId) {
        return this.choose(serviceId, (Object)null);
    }

    public ServiceInstance choose(String serviceId, Object hint) {
        Server server = this.getServer(this.getLoadBalancer(serviceId), hint);
        return server == null ? null : new RibbonLoadBalancerClient.RibbonServer(serviceId, server, this.isSecure(server, serviceId), this.serverIntrospector(serviceId).getMetadata(server));
    }

然后跟进CTRL this.getLoadBalancer(serviceId) 跟进得到的方法是

    protected ILoadBalancer getLoadBalancer(String serviceId) {
        return this.clientFactory.getLoadBalancer(serviceId);
    }

我们现在查看ILoadBalancer

public interface ILoadBalancer {
    void addServers(List<Server> var1);

    Server chooseServer(Object var1);

    void markServerDown(Server var1);

    /** @deprecated */
    @Deprecated
    List<Server> getServerList(boolean var1);

    List<Server> getReachableServers();

    List<Server> getAllServers();
}

我们猜测gelAllServers是获取所有可用服务列表的集合,现在我们查看一下其实现方法BaseLoadBalancer打一个断点查看一下

    public List<Server> getAllServers() {
        return Collections.unmodifiableList(this.allServerList);
    }

结果

springcloud 服务间通讯方式 Ribbon

这个刚好就是我们选择的PRODUCT的两个服务,就此我们可以判断,gelAllServers是根据服务名称获取服务列表的方法。

现在我们获取到服务列表了,下一步就应该根据规则选择一个服务进行返回。

5.3.3选择一个服务

现在跟进 this.getServer方法

    protected Server getServer(ILoadBalancer loadBalancer, Object hint) {
        return loadBalancer == null ? null : loadBalancer.chooseServer(hint != null ? hint : "default");
    }

然后跟进chooseServer方法

    public Server chooseServer(Object key) {
        if (this.counter == null) {
            this.counter = this.createCounter();
        }

        this.counter.increment();
        if (this.rule == null) {
            return null;
        } else {
            try {
                return this.rule.choose(key);
            } catch (Exception var3) {
                logger.warn("LoadBalancer [{}]:  Error choosing server for key {}", new Object[]{this.name, key, var3});
                return null;
            }
        }
    }

我们可以看见Server svr = this.rule.choose(key); 这个方法,这个就是根据规则选择一个服务

我们查看一下规则rule

  protected IRule rule;

构造函数

 this.rule = DEFAULT_RULE;

  private static final IRule DEFAULT_RULE = new RoundRobinRule();

通过名字我们可以看出来,负责均衡的方法规则是轮询的方式。

查看测试结果

第一次

springcloud 服务间通讯方式 Ribbon

第二次

springcloud 服务间通讯方式 Ribbon

第三次

springcloud 服务间通讯方式 Ribbon

我们跟进choose方法

   public Server choose(Object key) {
        ILoadBalancer lb = this.getLoadBalancer();
        Optional<Server> server = this.getPredicate().chooseRoundRobinAfterFiltering(lb.getAllServers(), key);
        return server.isPresent() ? (Server)server.get() : null;
    }

debug看到

lb变量里面的

springcloud 服务间通讯方式 Ribbon

我们可以看见所有服务列表,跟校验规则

然后根据

  Optional<Server> server = this.getPredicate().chooseRoundRobinAfterFiltering(lb.getAllServers(), key);

返回一个服务对象

springcloud 服务间通讯方式 Ribbon

5.4 修改规则

需要在application.yml

product: #访问服务名称
  ribbon:
    NFLoadBalancerRuleClassName: com.netflix.loadbalancer.RandomRule

规则在IRule的实现类里面可以选择,也可以自定义

springcloud 服务间通讯方式 Ribbon

本文同步分享在 博客“DencyCheng”(CSDN)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

点赞
收藏
评论区
推荐文章
刚刚好 刚刚好
2个月前
css问题
1、在IOS中图片不显示(给图片加了圆角或者img没有父级)<div<imgsrc""/</divdiv{width:20px;height:20px;borderradius:20px;overflow:h
blmius blmius
1年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
晴空闲云 晴空闲云
2个月前
css中box-sizing解放盒子实际宽高计算
我们知道传统的盒子模型,如果增加内边距padding和边框border,那么会撑大整个盒子,造成盒子的宽度不好计算,在实务中特别不方便。boxsizing可以设置盒模型的方式,可以很好的设置固定宽高的盒模型。盒子宽高计算假如我们设置如下盒子:宽度和高度均为200px,那么这会这个盒子实际的宽高就都是200px。但是当我们设置这个盒子的边框和内间距的时候,那
艾木酱 艾木酱
1个月前
快速入门|使用MemFire Cloud构建React Native应用程序
MemFireCloud是一款提供云数据库,用户可以创建云数据库,并对数据库进行管理,还可以对数据库进行备份操作。它还提供后端即服务,用户可以在1分钟内新建一个应用,使用自动生成的API和SDK,访问云数据库、对象存储、用户认证与授权等功能,可专
Wesley13 Wesley13
1年前
MySQL查询按照指定规则排序
1.按照指定(单个)字段排序selectfromtable_nameorderiddesc;2.按照指定(多个)字段排序selectfromtable_nameorderiddesc,statusdesc;3.按照指定字段和规则排序selec
Stella981 Stella981
1年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Stella981 Stella981
1年前
Angular material mat
IconIconNamematiconcode_add\_comment_addcommenticon<maticonadd\_comment</maticon_attach\_file_attachfileicon<maticonattach\_file</maticon_attach\
Wesley13 Wesley13
1年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
helloworld_28799839 helloworld_28799839
2个月前
常用知识整理
Javascript判断对象是否为空jsObject.keys(myObject).length0经常使用的三元运算我们经常遇到处理表格列状态字段如status的时候可以用到vue
helloworld_34035044 helloworld_34035044
5个月前
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为