在上一篇文章中介绍了Dubbo。
Dubbo是一个分布式服务框架,致力于提供高性能和透明化的RPC远程服务调用方案。它主要解决了在高访问量的情况下进程间的通讯问题,底层使用了socket来实现。我们知道消费者consumer调用服务者提供的服务时,因为两个程序不在同一个进程,所以不能够和同一个应用中那样直接调用,dubbo为我们处理好了通讯的问题。
当调用远程RPC服务时:
•首先,要解决通讯的问题,主要是通过在客户端和服务器之间建立TCP连接,远程过程调用的所有交换的数据都在这个连接里传输。连接可以是按需连接,调用结束后就断掉,也可以是长连接,多个远程过程调用共享同一个连接。
• 第二,要解决寻址的问题,也就是说,A服务器上的应用怎么告诉底层的RPC框架,如何连接到B服务器(如主机或IP地址)以及特定的端口,方法的名称名称是什么,这样才能完成调用。比如基于Web服务协议栈的RPC,就要提供一个endpoint URI,或者是从UDDI服务上查找。如果是RMI调用的话,还需要一个RMI Registry来注册服务的地址。
• 第三,当A服务器上的应用发起远程过程调用时,方法的参数需要通过底层的网络协议如TCP传递到B服务器,由于网络协议是基于二进制的,内存中的参数的值要序列化成二进制的形式,也就是序列化(Serialize)或编组(marshal),通过寻址和传输将序列化的二进制发送给B服务器。
• 第四,B服务器收到请求后,需要对参数进行反序列化(序列化的逆操作),恢复为内存中的表达方式,然后找到对应的方法(寻址的一部分)进行本地调用,然后得到返回值。
• 第五,返回值还要发送回服务器A上的应用,也要经过序列化的方式发送,服务器A接到后,再反序列化,恢复为内存中的表达方式,交给A服务器上的应用。
以上工作都由dubbo为我们做好了。所以我们可以直接使用服务提供者实现的业务类,就像和本地实现的一样。
Spring集成Dubbo非常简单。只需要简单的配置,Dubbo默认的是同步调用,当然也可以设置成异步调用。异步调用的方法可以参考这篇文章
Dubbo在代码实现过程中一般可以分为3个层次,一层是服务提供者provider,一层是消费者consumer,一层是service就是存接口,还包括一些参数传递所使用的JavaBean。一般可以将service打包成一个jar引入到provider与consumer中。
消费者相关配置:
消费者与服务提供者的bean的id可以不同,但是接口名必须相同。
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:dubbo="http://code.alibabatech.com/schema/dubbo"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://code.alibabatech.com/schema/dubbo
http://code.alibabatech.com/schema/dubbo/dubbo.xsd">
<!-- 提供方应用名称信息,这个相当于起一个名字,我们dubbo管理页面比较清晰是哪个应用暴露出来的 -->
<dubbo:application name="dubbo-consumer" logger="log4j"/>
<!-- 使用zookeeper注册中心暴露服务地址 -->
<!-- check 启动时检查提供者是否存在,true报错,false忽略 -->
<!-- registry 从指定注册中心注册获取服务列表,在多个注册中心时使用,
值为<dubbo:registry>的id属性,多个注册中心ID用逗号分隔 -->
<!-- subscribe是否向此注册中心订阅服务,如果设为false,将只注册,不订阅 -->
<dubbo:registry id="dubboConsumer" address="${zookeeper}" file="${registry.file}" check="${dubbo.check}"
subscribe="${dubbo.subscribe}" register=""/>
<dubbo:monitor protocol="registry"/>
<!-- 服务消费方 -->
<dubbo:consumer timeout="30000" check="${dubbo.check}" retries="0"/>
<!-- 生成远程服务代理,可以和本地bean一样使用demoService -->
<dubbo:reference id="demoService" interface="com.xfl.provider.service.IDemoService" registry="dubboConsumer"
check="${dubbo.check}" version="1.0.0"/>
</beans>
在业务代码中像本地的bean一样使用
@RestController
@RequestMapping("/Test")
public class TestController extends BaseController {
/**
* 该服务由dubbo服务提供者实现,也可以像本地bean一样的使用.
*/
@Resource
private IDemoService demoService;
/**
* 日志打印.
*/
private final Logger log = LoggerFactory.getLogger(TestController.class);
/**
* 注入Service.
*/
@Resource
private ITestService testService;
/**
* 用于测试.
*
* @return 返回测试结果
*/
@RequestMapping(method = RequestMethod.GET)
public Response test() {
log.info("Test");
System.out.println(demoService.sayHello("String"));
return new Response(ResponseStatusEnum.SUCCESS.getCode(), testService.test());
}
}
service提供的存接口
/**
*
* @author XFL.
* @time 2016年10月23日 上午10:16:39
* @description dubbo测试例子
*/
public interface IDemoService {
/**
* 测试样例.
* @param name 测试参数name
* @return 返回测试结果
* @return String 返回类型字符串
* @todo TODO 测试样例
*/
String sayHello(String name);
}
provider服务提供者相关配置以及代码
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:dubbo="http://code.alibabatech.com/schema/dubbo"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://code.alibabatech.com/schema/dubbo
http://code.alibabatech.com/schema/dubbo/dubbo.xsd">
<!-- 提供方应用名称信息,这个相当于起一个名字,我们dubbo管理页面比较清晰是哪个应用暴露出来的 -->
<dubbo:application name="dubbo-provider" logger="log4j"/>
<!-- 使用zookeeper注册中心暴露服务地址 -->
<!-- check 启动时检查提供者是否存在,true报错,false忽略 -->
<!-- registry 从指定注册中心注册获取服务列表,在多个注册中心时使用,
值为<dubbo:registry>的id属性,多个注册中心ID用逗号分隔 -->
<!-- subscribe是否向此注册中心订阅服务,如果设为false,将只注册,不订阅 -->
<dubbo:registry id="dubboProvider" address="${zookeeper}" file="${registry.file}" check="${dubbo.check}"
subscribe="${dubbo.subscribe}" register=""/>
<dubbo:monitor protocol="registry"/>
<!-- 服务消费方 -->
<dubbo:consumer timeout="30000" check="${dubbo.check}" retries="0"/>
<!-- 和本地bean一样实现服务 -->
<bean id="demoService" class="com.xfl.provider.impl.DemoServiceImpl" />
<!-- 要暴露的服务接口 -->
<dubbo:service interface="com.xfl.provider.service.IDemoService" registry="dubboProvider"
ref="demoService" version="1.0.0"/>
</beans>
在服务提供者中实现该服务
/**
* Created by XFL.
* time on 2016/10/26 21:20
* description: dubbo服务实现类
*/
public class DemoServiceImpl implements IDemoService {
/**
* 测试样例.
*
* @param name 测试参数name
* @return String 返回类型字符串
*/
@Override
public String sayHello(String name) {
return name + "Hello!";
}
}
点击这里可以下载consumer代码,在GitHub上
点击这里可以下载Provider代码,在GitHub上
点击这里下载zookeeper Windows 位版本
参考资料:
http://blog.youkuaiyun.com/flashflight/article/details/43939275
http://sanwen8.cn/p/1abLjG6.html