Dubbo源码学习三--Dubbo源码结构及实现方式

Dubbo源码结构

Dubbo源码目录结构.png

在github下载Dubbo源码之后导入本地开发工具,自己用的是idea,导入之后可以看到Dubbo源码的目录结构,主要的几个目录及该目录的作用如下:

dubbo-cluster: 集群容错模块,包含负载均衡策略,集群容错策略以及路由等   
dubbo-common:通用逻辑模块,提供工具类和通用模型
dubbo-compatible:兼容性模块
dubbo-config:配置模块,主要实现API配置,属性配置,XML配置等工功能
dubbo-configcenter:动态配置中心
dubbo-container:容器模块,没有使用Tomcat等Web容器,而是使用Main方法加载Spring容器。
dubbo-demo:提供了三种方式的远程调用示例。
dubbo-monitor:监控模块,主要监控接口调用的次数、时间等信息。
dubbo-registry: 注册中心模块。
dubbo-rpc: 抽象各种通信协议以及动态代理(容易和remoting模块相混淆)
dubbo-remoting:远程通信模块,为消费者和生产者提供远程通信能力。
dubbo-plugin:插件模块。

使用Dubbo框架的几种方式:

dubbo-demo模块提供了几种使用Dubbo框架进行远程调用的方式

使用XML配置的方式:

 使用Dubbo实现远程调用需要满足的几个条件:
  1.服务提供者
  2.服务调用者
  3.注册中心(默认使用zookeeper作为注册中心,需要在本机电脑安装zookeeper并且启动zookeeper服务)。

1.服务提供者

首先创建服务提供者,可以看下示例中项目的结构:


Dubbo服务提供者结构.png

dubbo-provider.xml文件配置了一些注册中心地址,暴露接口等信息。示例中的xml配置文件详解。

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      xmlns:dubbo="http://dubbo.apache.org/schema/dubbo"
      xmlns="http://www.springframework.org/schema/beans"
      xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.3.xsd
      http://dubbo.apache.org/schema/dubbo http://dubbo.apache.org/schema/dubbo/dubbo.xsd">
   <!--服务名称-->
   <dubbo:application name="demo-provider"/>
   <!--注册中心地址-->
   <dubbo:registry address="zookeeper://127.0.0.1:2181"/>
   <!--远程调用协议-->
   <dubbo:protocol name="dubbo"/>
   <!--注册beam-->
   <bean id="demoService" class="org.apache.dubbo.demo.provider.DemoServiceImpl"/>
   <!--暴露提供的服务-->
   <dubbo:service interface="org.apache.dubbo.demo.DemoService" ref="demoService"/>
</beans>

Application类代码:

package org.apache.dubbo.demo.provider;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Application {
    public static void main(String[] args) throws Exception {
        /**加载配置文件*/
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring/dubbo-provider.xml");
        /**启动容器*/
        context.start();
        System.in.read();
    }
}

再看下具体暴露的接口的实现DemoServiceImpl:


package org.apache.dubbo.demo.provider;

import org.apache.dubbo.demo.DemoService;
import org.apache.dubbo.rpc.RpcContext;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class DemoServiceImpl implements DemoService {
   private static final Logger logger = LoggerFactory.getLogger(DemoServiceImpl.class);

   @Override
   public String sayHello(String name) {
       logger.info("Hello " + name + ", request from consumer: " + RpcContext.getContext().getRemoteAddress());
       return "Hello " + name + ", response from provider: " + RpcContext.getContext().getLocalAddress();
   }
}

主要提供了接口的实现,实现逻辑和简单,主要获取远程调用该方法的地址以及返回一些字符串信息。

2.服务调用者

Dubbo服务调用者结构.png

服务调用者对于dubbo相关信息的配置在dubbo-consumer.xml配置文件中,一下是该配置文件配置信息的详解:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
     xmlns:dubbo="http://dubbo.apache.org/schema/dubbo"
     xmlns="http://www.springframework.org/schema/beans"
     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.3.xsd
     http://dubbo.apache.org/schema/dubbo http://dubbo.apache.org/schema/dubbo/dubbo.xsd">
  <!--应用名称-->
  <dubbo:application name="demo-consumer"/>
  <!--注册中心地址-->
  <dubbo:registry address="zookeeper://127.0.0.1:2181"/>
  <!--定义reference,注意id的值一定要和dubbo-provider.xml中配置的bean的id相同-->
  <dubbo:reference id="demoService" check="false" interface="org.apache.dubbo.demo.DemoService"/>
</beans>

Consumer调用远程方法的实现:


package org.apache.dubbo.demo.consumer;
import org.apache.dubbo.demo.DemoService;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Application {
    public static void main(String[] args) {
        /**加载配置文件*/
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring/dubbo-consumer.xml");
        context.start();
        /**获取接口实例*/
        DemoService demoService = context.getBean("demoService", DemoService.class);
        /**调用远程服务*/
        String hello = demoService.sayHello("world");
        System.out.println("result: " + hello);
    }
}

总结:
生产者其实是demoService接口的一种实现,消费者调用demoService接口中的方法,Dubbo封装成远程调用,用户无感知,可以像调用本地接口一样调用接口的远程实现。

3.服务之间的调用

1.启动zookeeper中间件

mac笔记本安装好zookeeper之后进入zookeper的bin目录,输入./zkServer.sh start 指令即可启动zookeeper中间件。一下为几篇各个操作系统下安装zookeeper单机版的教程:
windows系统安装zookeeper教程
mac系统安装zookeeper教程
linux系统安装zookeeper教程

启动zookeeper之后进入日志路径查看启动日志:


zookeeper启动日志.png

2.启动服务提供者:

服务启动者启动时的日志如下:

生产者启动日志1.png

生产者启动日志2.png

有几个关键的步骤:

 1.加载配置信息完成
 2.初始化连接zookeeper客户端
 3.查看缓存中是否有连接zookeeper的客户端,没有客户端则创建客户端
 4.导出本地服务和远程服务,并且打印出了本地服务和远程服务的url地址
 5.将导出的地址注册到注册中心

服务提供者启动之后zookeeper日志

服务提供者启动后zookeeper日志.png

Zookeeper日志主要展示了一下几个信息:

1.接收到来自服务提供者的socket连接请求
2.尝试创建session
3.建立连接

3.启动调用者

消费者启动过程.png

通过日志输出可以看到关键的几个步骤:

1.查找zookeeper缓存文件,没有则创建新的client连接zookeeper
2.向zookeeper进行注册
3.通过Netty客户端连接远程服务,并且调用远程服务

总结:通过生产者和消费者的调用过程大致了解了通过XML配置方式进行远程调用的一个过程,具体调用过程中Dubbo将提供者的地址暴露给zookeeper,调用者怎么通过Dubbo调用的远程服务,需要深入Dubbo的源码进行分析。

使用Dubbo框架的其他两种方式

除了xml的方式,Dubbo还提供了另外两种使用方式,可以运行实例分析其各种实现方式的特点。

我是割草的小猪头,不断学习,不断进步,后续陆续更新Dubbo系列的文章,如您有兴趣一起了解,欢迎关注,如文章中有不妥之处,欢迎指正!

Dubbo系列文章一--Dubbo重点掌握模块
Dubbo系列文章二--配置文件加载过程

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,290评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,399评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,021评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,034评论 0 207
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,412评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,651评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,902评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,605评论 0 199
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,339评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,586评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,076评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,400评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,060评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,083评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,851评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,685评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,595评论 2 270

推荐阅读更多精彩内容