1、系统间通信
无论是soa(面向服务架构)还是分布式架构,每个独立的业务之间都需要系统间通信。比如实现商品列表查询需要两个系统之间通信。系统之间的远程通信有几种方式呢?
1.1、Webservice:效率不高,基于soap协议。不推荐使用。
1.2、restful:基于http+json。很多项目中使用,但是如果服务太多的话,服务间调用混乱,需要治疗。
1.3、dubbo:使用rpc协议进行远程调用,直接使用socket通信。传输效率高,并且可以统计系统之间的调用关系、调用次数。
2、什么是dubbo
文档 http://dubbo.io/User+Guide-zh.htm
dubbo就是资源调度和治理中心的管理工具。
dubbo采用全spring配置方式,透明化接入应用,对应用没有任何api侵入,只需要spring加载dubbo的配置即可,dubbo基于spring的schema进行扩展。
dubbo扩展了spring schema定义,两条定义如下:
xmlns:dubbo="http://code.alibabatech.com/schema/dubbo"
http://code.alibabatech.com/schema/dubbo http://code.alibabatech.com/schema/dubbo/dubbo.xsd
dubbo schema定义了如下对象:
- ApplicationConfig
- ModuleConfig
- RegistryConfig
- MonitorConfig
- ProviderConfig
- ConsumerConfig
- ProtocolConfig
- ServiceBean
- ReferenceBean
- AnnotationBean
在spring xml文件中按照下面的方式注入dubbo schema中定义的对象:
<dubbo:application name="hello-world-app" />
<dubbo:registry address="multicast://224.5.6.7:1234" />
<dubbo:protocol name="dubbo" port="20880" />
<dubbo:service interface="com.alibaba.dubbo.demo.DemoService" ref="demoServiceLocal" />
<dubbo:reference id="demoServiceRemote" interface="com.alibaba.dubbo.demo.DemoService" />
<dubbo:annotation package="com.alibaba.dubbo.config.spring.annotation.consumer" />
3、dubbo的架构
- Provider: 暴露服务的服务提供方
- Consumer: 调用远程服务的服务消费方
- Registry: 服务注册与发现的注册中心(zookeeper)
- Monitor: 统计服务的调用次数和调用时间
- Container: 服务运行容器
4、dubbo的使用
4.1、安装注册中心zookeeper
注册中心负责服务地址的注册和查找,相当于目录服务,服务提供者与消费者只在注册中心启动与注册中心交互,注册中心不转发请求。zookeeper是Apache Hadoop的一个子项目,是一个树形的目录服务,支持变更推送,所以推荐使用。安装步骤如下:
1、安装jdk
2、解压缩zookeeper包
3、将conf文件夹下zoo_sample.cfg复制一份改名为zoo.cfg
4、修改dataDir属性,指定一个真实路径
5、启动zookeeper:bin/zkServer.sh start
关闭zookeeper:bin/zkServer.sh stop
查看zookeeper状态:bin/zkServer.sh status
4.2、spring配置及服务发布与引用
- 定义服务接口: (该接口需单独打包,在服务提供方和消费方共享)
- 在服务提供方实现接口:(对服务消费方隐藏实现)
- 用Spring配置声明暴露服务
- 加载提供方Spring配置
- 通过Spring配置引用远程服务
- 加载消费方Spring配置,并调用远程服务(也可以使用IoC注入)
服务提供方暴露服务:
放在service工程中,例如:taotao-manager-service,taotao-content-service
- 需要注册地址。
- 需要暴露服务端口。端口决定不同的服务,即每个服务的端口是不一样的。
- 需要声明服务接口,关键字是dubbo:service ,“interface”是interface工程中定义的服务接口,“ref”是service工程中服务提供方实现的接口。
<!-- 发布dubbo服务 -->
<!-- 提供方应用信息,用于计算依赖关系 -->
<dubbo:application name="taotao-manager" />
<!-- 注册中心的地址 -->
<dubbo:registry protocol="zookeeper" address="192.168.25.167:2181" />
<!-- 用dubbo协议在20880端口暴露服务 -->
<dubbo:protocol name="dubbo" port="20880" />
<!-- 声明需要暴露的服务接口 -->
<dubbo:service interface="com.taotao.service.ItemService" ref="itemServiceImpl" timeout="300000"/>
<dubbo:service interface="com.taotao.service.ItemCatService" ref="itemCatServiceImpl" timeout="300000"/>
加载提供方spring配置
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext(new String[] {"http://10.20.160.198/wiki/display/dubbo/provider.xml"});
context.start();
服务消费方引用服务:
放在web工程中,例如taotao-manager-web,taotao-item-web
- 需要注册地址。
- 需要引用服务接口,关键字是dubbo:reference,“interface”是interface工程中定义的服务接口,“id”一般为首字母小写的对应接口类名称。
<!-- 引用dubbo服务 -->
<dubbo:application name="taotao-manager-web"/>
<dubbo:registry protocol="zookeeper" address="192.168.25.167:2181"/>
<dubbo:reference interface="com.taotao.service.ItemService" id="itemService" />
<dubbo:reference interface="com.taotao.service.ItemCatService" id="itemCatService" />
<dubbo:reference interface="com.taotao.content.service.ContentCategoryService" id="contentCategoryService" />
<dubbo:reference interface="com.taotao.content.service.ContentService" id="contentService" />
<dubbo:reference interface="com.taotao.search.service.SearchItemService" id="searchItemService" />
加载消费方spring配置
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext(
new String[] { "http://10.20.160.198/wiki/display/dubbo/consumer.xml" });
context.start();
DemoService demoService = (DemoService) context.getBean("demoService"); // 获取远程服务代理
String hello = demoService.sayHello("world"); // 执行远程方法
System.out.println(hello); // 显示调用结果