Dubbo的服务消费主要包括两个部分。第一大步是ReferenceConfig类的init方法调用Protocolrefer方法生成Invoker实例,这是服务消息的关键。第二大步是把Invoker通过动态代理转换成实现用户接口的动态代理引用。这里的Invoker承载了网络连接、服务调用和重试等功能。

服务暴露起点

在消费者的配置文件中存在这个代码:

<!-- 生成远程服务代理,可以和本地bean一样使用demoService -->
<dubbo:reference id="demoService" interface="org.apache.dubbo.demo.DemoService" />

它会生成一个ReferenceBean,实现了FactoryBean接口,继承了ReferenceConfig,所以ReferenceBean作为dubbo中能够生产对象的工厂Bean,而我们要引用服务,也就要有一个该服务的对象。

public class ReferenceBean<T> extends ReferenceConfig<T> implements FactoryBean,
ApplicationContextAware, InitializingBean, DisposableBean {

服务引用被触发有两个时机:

  • Spring容器调用ReferenceBean的afterPropertiesSet方法时引用服务(饿汉式)
  • 在ReferenceBean对应的服务被注入到其他类中时引用(懒汉式)

默认情况下,Dubbo使用懒汉式引用服务。如果需要使用饿汉式,可通过配置dubbo:reference的init属性开启。

因为ReferenceBean实现了FactoryBean接口的getObject()方法,所以在加载bean的时候,会调用ReferenceBean的getObject()方法。

ReferenceBean.getObject()  --->  ReferenceConfig.get()   --> ReferenceConfig.init()

在init()方法中主要有这几步:

(1) 检测本地存根和mock合法性

(2) 添加协议版本、发布版本、时间戳、application、module、consumer、protocol等所有信息到map中。

(3) 单独处理方法配置,设置重试次数配置以及设置该方法对异步配置信息。

(4) 添加消费者ip地址到map

(5)创建代理对象,调用ReferenceConfig.createProxy()方法。

(6) 生成ConsumerMpdel存入到ApplicationModel中。

之后介绍ReferenceConfig.createProxy()方法。主要有下面几步:

  • 如果是本地调用,则直接使用InjvmProtocol的refer方法生成Invoker实例。
  • 如果不是本地调用,但是选择直连的方式进行调用,则分割配置的多个url。如果协议是配置registry,则表明用户想使用指定的注册中心,配置url后将url保存到urls里面,否则就合并url,并且保存到urls。
  • 如果是通过注册中心来进行调用,则先校验所有的注册中心,然后假加载注册中心的url,遍历每个url,加入监控中心url配置,最后把每个url保存到urls。
  • 如果urls的个数是1,是单注册中心,直接引用RegistryProtocol的refer构建Invoker实例;如果urls的数量大于1,说明是多注册中心,则对每个url都生成Invoker,利用cluster.join()方法将多个Invoker进行合并成一个Invoker。
  • 最后调用proxyFactory.getProxy(invoker)方法。

然后介绍RegistryProtocol.refer(Class<T> type, URL url)方法生成invoker。

如果是注册中心服务,则直接返回注册中心服务的invoker;如果不是,则先处理组配置,根据组配置来决定Cluster的实现方式,如果有多个组,则使用MergeableCluster,然后调用doRefer(Cluster, Registry, Class, URL)方法。

然后介绍doRefer()方法。

(1) 创建一个RegistryDirectory实例,设置注册中心、协议等信息

(2) 生成服务消费者链接。

(3) 注册消费信息到注册中心。

(4) 订阅该服务下的providers、configurators、routers等节点下的数据。完成订阅后,RegistryDirectory会受到这几个节点下的子节点信息。

(5) 由于一个服务可能部署在多台服务器上,这样就会在providers产生多个节点,这个时候就需要Cluster将多个服务节点合并成一个,并生成一个Invoker.

在RegistryDirectory实现了NotifyListener接口,服务变更会触发这个类回调notify方法,用于重新引用服务。当发起订阅请求时会进行一次数据拉取操作,同时触发RegistryDirectory.nofity()方法。这是会执行toInvokers()方法进行Invoker转换。

(1) 根据消费者protocol配置过滤不匹配的协议。

(2) 合并provider端配置数据,比如服务端IP和port等。

(3) 忽略重复推送的服务列表

(4) 使用具体协议创建远程连接,new InvokerDelegate<T>(protocol.refer(serviceType, url), url, providerUrl)

具体的Invoker创建是在DubboProtocol.refer()中实现。Dubbo协议在返回DubboInvoker对象之前会初始化客户端连接对象。

调用DubboProtocol.initClient()方法 -> Exchangers.connect()方法,根据SPI机制加载HeaderExchangeClent,调用connect()方法。然后调用Transporter类的connect()方法,默认是NettyTransporter类。

public class DubboInvoker<T> extends AbstractInvoker<T> {
private final ExchangeClient[] clients;
private final AtomicPositiveInteger index = new AtomicPositiveInteger();
private final String version;
private final ReentrantLock destroyLock = new ReentrantLock();
private final Set<Invoker<?>> invokers;
public abstract class AbstractInvoker<T> implements Invoker<T> {
protected final Logger logger = LoggerFactory.getLogger(getClass());
private final Class<T> type;
private final URL url;
private final Map<String, Object> attachment;
private volatile boolean available = true;
private AtomicBoolean destroyed = new AtomicBoolean(false);

Dubbo之服务消费的更多相关文章

  1. Dubbo之服务消费原理

    前言 上篇文章<Dubbo之服务暴露>分析 Dubbo 服务是如何暴露的,本文接着分析 Dubbo 服务的消费流程.主要从以下几个方面进行分析:注册中心的暴露:通过注册中心进行服务消费通知 ...

  2. 使用dubbo分布式服务框架发布服务及消费服务

    什么是DUBBO DUBBO是一个分布式服务框架,致力于提供高性能和透明化的RPC远程服务调用方案. 准备工作 安装zookeeper ZooKeeper是一个分布式的,开放源码的分布式应用程序协调服 ...

  3. dubbo注册服务和消费服务---入门篇

    本文介绍如何用dubbo+zk来实现一个注册服务 + 消费服务的入门小demo 需要环境:zk服务器 两个maven项目,一个负责提供服务,一个负责消费服务. dubbo-service 服务端 po ...

  4. Dubbo学习笔记10:Dubbo服务消费方启动流程源码分析

    同理我们看下服务消费端启动流程时序图: 在<Dubbo整体架构分析>一文中,我们提到服务消费方需要使用ReferenceConfig API来消费服务,具体是调用代码(1)get()方法来 ...

  5. Dubbo学习笔记4:服务消费端泛化调用与异步调用

    本文借用dubbo.learn的Dubbo API方式来解释原理. 服务消费端泛化调用 前面我们讲解到,基于Spring和基于Dubbo API方式搭建简单的分布式系统时,服务消费端引入了一个SDK二 ...

  6. Spring Cloud Alibaba(四)实现Dubbo服务消费

    本项目演示如何使用 Spring Cloud Alibaba 完成 Dubbo 的RPC调用. Spring Cloud与Dubbo Spring Cloud是一套完整的微服务架构方案 Dubbo是国 ...

  7. dubbo入门之服务消费

    今天,我们来看看dubbo消费的执行过程 首先,我们都知道dubbo是一个基于netty实现的RPC框架,底层通信是使用netty来实现的.在学习dubbo的时候,或许我们都会有下面的这些疑惑: 1. ...

  8. 源码分析Dubbo服务消费端启动流程

    通过前面文章详解,我们知道Dubbo服务消费者标签dubbo:reference最终会在Spring容器中创建一个对应的ReferenceBean实例,而ReferenceBean实现了Spring生 ...

  9. Dubbo分布式服务框架入门

    参考http://blog.csdn.net/u013142781/article/details/50387583 一.Dubbo概念介绍 1.1.Dubbo是什么? Dubbo是一个分布式服务框架 ...

随机推荐

  1. Android 6.0(棉花糖)新特性

    1.支持4K显示 Android 6.0本身已经支持4K显示,会通过一定优化形式使4K内容更加清晰. 2. 启动验证 (更完整的应用权限管理) Android 6.0在开机时会自动运行验证代码,检测设 ...

  2. ORB-SLAM2 论文&代码学习 —— LoopClosing 线程

    转载请注明出处,谢谢 原创作者:Mingrui 原创链接:https://www.cnblogs.com/MingruiYu/p/12369339.html 本文要点: ORB-SLAM2 LoopC ...

  3. 常见CSS3选择器和文本字体样式汇总

    常见的CSS3选择器包含:常用基本.属性.伪类.层级(组合)选择器,具体使用情况建议先阅读css选择器四大类:基本.组合.属性.伪类对于选择器的使用有一个基本了解,选择器的作用在于通过它找到元素,并且 ...

  4. TampeMonkey 关于 youtube的两个插件

    一个是 Video Speed Buttons 负责调速 一个是 YouTube Links  负责下载不同分辨率的视频

  5. MySQL基础(7) | 触发器

    MySQL基础(7) | 触发器 基本语法 创建 CREATE TRIGGER trigger_name trigger_time trigger_event ON table_name FOR EA ...

  6. 【DTOJ】2704:数字互换

    DTOJ 2704:数字互换  解题报告 2017.11.11 第一版 ——由翱翔的逗比w原创 题目信息: 题目描述 输入两个数作为交换数,输出已交换顺序后的两个值. 输入 两个整数,空格隔开 输出 ...

  7. Eclipse jee最新版国内镜像点下载方式

    Eclipse jee是Eclipse的JavaEE版本,官网服务器在国外,无条件下载的可以使用国内镜像点 清华大学网站镜像网站资料,可用于下载其它版本的Eclipsehttps://mirrors. ...

  8. A tiny problem with integers

    # A tiny problem with integers 给定长度为N的数列A,然后输入M行操作指令. 第一类指令形如“C l r d”,表示把数列中第l~r个数都加d. 第二类指令形如“Q X” ...

  9. 牛客网剑指offer第13题——调整数组顺序使得奇数位于偶数前面

    题目来源:剑指offer 题目: 输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于数组的后半部分,并保证奇数和奇数,偶数和偶数之间的相对位置不变 ...

  10. HTML指定页面编码

    HTML指定页面编码 <meta charset="UTF-8" />