java开发分布式服务框架Dubbo暴露服务过程详解
作者:又蠢又笨的懒羊羊程序猿 发布时间:2021-07-19 17:59:50
Dubbo服务暴露机制
前言
在进行服务暴露机制的分析之前,必须谈谈什么是URL
,在Dubbo服务暴露过程中URL
是无处不在的,贯穿了整个过程。
一般情况下,URL
指的是统一资源定位符,标准格式如下:
protocol://host:port/path?key1=value1&key2=value2
Dubbo就是用这种URL
的方式来作为约定的参数类型,服务之间也是用URL
来进行交互。
Dubbo用URL
作为配置总线,贯穿整个源码体系。
URL
的具体参数如下:
protocol
:dubbo中的各种协议,如:dubbo、http、thrift
host/port
:主机/端口号
path
:接口名称
parameters
:参数键值对
服务暴露流程
大体流程分为三个步骤:
检测配置,有些配置为空值会有默认值自动填充并自动组装为URL
暴露服务,包括暴露到本地和远程的服务注册为服务到服务注册中心
源码解析
先来到ServiceBean
中
可以看到这个类实现了ApplicationListener<ContextRefreshedEvent>
,这样就会在Spring IOC容器刷新之后调用onApplicationEvent
方法,这个方法中做的就是服务暴露。
从if
判断条件中,我们可以看到如果配置中没有设置延迟暴露、并且没有暴露过、并且支持暴露的情况下就会执行this.export()
,最终会调用父类ServiceConfig
中的export
方法。
上述方法中主要就是进行配置检查,然后进入this.doExport()
进行服务暴露,这个方法前面的部分都是一些配置检查,重点放在最后的this.doExportUrls()
。
从上述源码中可以看到Dubbo是支持多注册中心的,并且支持多种协议,一个服务如果有多个协议那么都需要暴露。this.loadRegistries()
是根据配置组装成注册中心相关的URL,如下:
registry://127.0.0.1:2181/com.alibaba.dubbo.registry.RegistryService?application=user-service-provider-one&dubbo=2.6.2&pid=29056®istry=zookeeper×tamp=1632309180780
接下来进入this.doExportUrlsFor1Protocol()
,
构建完成之后的URL如下图所示:
使用的是dubbo协议,接下来要做的就是根据URL来进行服务暴露
本地暴露
进入exportLocal()
看一下,这个方法做的事就是进行本地暴露,使用的是injvm
协议,这个方法中对URL进行修改,将协议改为injvm
。
此处的protocol.export()
是标注了@Adaptive
注解的,设计自适应拓展,调用该方法之后会生成代理类,代理类通过URL中的参数得知所使用的协议,然后通过Dubbo SPI机制实现拓展,选择对应的实现类进行export()
方法,此处的方法会选择InjvmProtocol
中的export()
进行暴露。
经过转化后得到的export
如下所示:
如图所示,具体实现类被层层包装。
为什么要封装成invoker呢?
封装成invoker是为了屏蔽调用的细节,统一暴露出一个可执行体,使得调用者调用时变得便捷简易。
为什么要使用本地暴露呢?
因为可能存在本地不同服务之间相互调用的情况,因此本地暴露后,内部可以直接进行调用,而省去了不必要的网络通信。
此处引用一下博主敖丙的文章中的本地暴露流程图:
远程暴露
此时本地暴露已经完成了,现在开始远程暴露。
流程与本地暴露相似,都需要先封装为invoker然后再进一步生成exporter,以下是registryURL.addParameterAndEncoded()
方法将URL进行拼接后的结果:
可以看到先是走registry
协议,然后参数中有一个export=dubbo://
,此时会走dubbo
协议,所以我们得知SPI会先通过registry
协议找到RegistryPotocol
然后执行export()
,然后再执行DubboPotocol
中的export()
。
现在我们进入RegistryPotocol
中的export()
方法。
上述操作主要是讲export转化为exporter,获取注册中心的相关配置,如果需要注册则向注册中心进行注册,并且在ProviderConsumerRegTable
中记录服务提供者信息,进入源码中,我们发现实质就是往一个ConcurrentHashMap
中塞入invoker,key是服务接口全类名,value是一个set,set中存储包装过的invoker。
然后进入 doLocalExport
方法内部
箭头指向的方法最终会调用DubboProtocol
中的export()
需要创建server
上述代码中,主要就是根据URL创建出key,然后将key和invoker存入exporterMap中关联起来,如果是初次服务暴露则需要创建监听服务器,默认是NettyServer,并且初始化各种Handler,比如心跳检测,解码等。
至此服务暴露流程结束。
此处引用博主敖丙的文章中的本地暴露流程图:
如果不足或错误欢迎评论指正。
来源:https://blog.csdn.net/TaylorSwiftiiln/article/details/120444877


猜你喜欢
- 我有一个文本输入对话框,当我点击对话框上的“是”按钮,它会验证输入,然后关闭对话框。但是,如果输入错误,我想停留在同一个对话框中.。每一次,
- 本文实例讲述了java基于递归算法实现汉诺塔问题。分享给大家供大家参考,具体如下:package test;import java.util
- 以前使用spring的使用要注入property要配置PropertyPlaceholder的bean对象。在springboot除&nbs
- 在 C# 中,结构体是值类型数据结构。它使得一个单一变量可以存储各种数据类型的相关数据。struct 关键字用于创建结构体。结构体是用来代表
- Maven 作为一个优秀的项目管理工具,其插件机制为其功能扩展提供了非常大的便捷性。虽然说大多数情况下,我们可能不太会自己去编写 Maven
- 本文实例讲述了Android编程之短信列表的时间显示。分享给大家供大家参考,具体如下:Android的短信的时间的显示做的很精细,首先保存在
- 引言相信伙伴们在日常的开发工作中,一定会遇到事件冲突的问题,e.g. 一个页面当手指滑动的时候,会翻到下一页;点击的时候,需要响应页面中的元
- Java的反射机制允许我们对一个类的加载、实例化、调用方法、操作属性的时期改为在运行期进行,这大大提高了代码的灵活度。但在运行期进行反射操作
- 本文实例为大家分享了C#实现学生档案查询的具体代码,供大家参考,具体内容如下using System;using System.Collec
- AndroidStudio 实现加载字体资源的方法在android中字体的格式总是不能尽善尽美的显示出来 , 于是要求我们使用一
- //初始化propertiesProperties pro = new Properties();try { &nbs
- 问题换了台开发机,重新安装了下开发环境。突然发现Visual Studio Code无法用来调试Unity了。明明流程都是按照Unity官方
- 本文实例为大家分享了Android仿QQ讨论组头像展示的具体代码,供大家参考,具体内容如下一、效果图二、实现基本实现过程:1.将原图片读取为
- 传统方式克隆羊问题现在有一只羊 tom,姓名为: tom,年龄为:1,颜色为:白色,请编写程序创建和 tom羊属性完全相同的10只羊。传统方
- 前言今天小编给大家带来的是用代码一键实现银行卡识别与绑定功能。银行卡识别的应用场景 介绍开发步骤前,我们先来谈谈银行卡识别的具体应用场景,
- 概述Spring Cloud中,客户端的负载均衡使用的是Ribbon,Ribbon的超时时间默认很短,需要进行调整。Spring Cloud
- 一、案例场景遇到过这样的场景,在定义一个static修饰的Map时,使用了大量的put()方法赋值,就类似这样——public static
- Parallel类(https://www.jb51.net/article/244267.htm)的并行任务需要结束后才能运行后面的代码,
- 方法一:简单的方法就是改变picturebox 控件的borderstyle样式 currentSelectPicBox.Bord
- Groovy 简介Groovy 是构建在 JVM 上的一个轻量级却强大的动态语言,它结合了 Python、Ruby 和 Smalltalk