1. Dubbo是什么
阿里推出的分布式服务框架,是一个基于SOA(面向服务的架构)的基础设施,致力于提供高性能和透明化的RPC远程服务调用方案,以及SOA服务治理方案。
优点:
远程通讯:提供对多种基于长连接的NIO框架抽象封装,包括多种线程模型,序列化,以及“请求-响应”模式的信息交换方式。
集群容错:提供基于接口方法的透明远程过程调用,包括多协议支持,以及软负载均衡,失败容错,地址路由,动态配置等集群支持。
自动发现: 基于注册中心目录服务,使服务消费方能动态的查找服务提供方,使地址透明,使服务提供方可以平滑增加或减少机器。
Dubbo采用全Spring配置方式,透明化接入应用,对应用没有任何API侵入,只需用Spring加载Dubbo的配置即可。
2. 应用场景
需要用到分布式的场景:
当网站变大后,不可避免的需要拆分应用进行服务化,以提高开发效率,调优性能,节省关键竞争资源等。
当服务越来越多时,服务的URL地址信息就会爆炸式增长,配置管理变得非常困难,F5硬件负载均衡器的单点压力也越来越大。
当进一步发展,服务间依赖关系变得错踪复杂,甚至分不清哪个应用要在哪个应用之前启动,架构师都不能完整的描述应用的架构关系。
接着,服务的调用量越来越大,服务的容量问题就暴露出来,这个服务需要多少机器支撑?什么时候该加机器?等等……
3. 架构
Container: 服务运行容器
Provider: 暴露服务的服务提供方。
Consumer: 调用远程服务的服务消费方。
Registry: 服务注册与发现的注册中心。
Monitor: 统计服务的调用次调和调用时间的监控中心。
0. 服务容器负责启动,加载,运行服务提供者。
1. 服务提供者在启动时,向注册中心注册自己提供的服务。
2. 服务消费者在启动时,向注册中心订阅自己所需的服务。
3. 注册中心返回服务提供者地址列表给消费者,如果有变更,注册中心将基于长连接推送变更数据给消费者。
4. 服务消费者,从提供者地址列表中,基于软负载均衡算法,选一台提供者进行调用,如果调用失败,再选另一台调用。
5. 服务消费者和提供者,在内存中累计调用次数和调用时间,定时每分钟发送一次统计数据到监控中心。
4. dubbo注册中心
通过将服务统一的管理起来,可有效优化内部应用对服务发布/使用的流程和管理。
服务注册中心可以通过特定的协议来完成服务对外的统一。
dubbo提供的注册中心类型:
- Multicast注册中心
- Zookeeper注册中心
- Redis注册中心
- Simple注册中心
Zookeeper:
它是一个为分布式应用提供一致性服务的软件,提供的功能包括:配置维护、域名服务、分布式同步、组服务等。
是树型的目录服务的数据存储,能做到集群管理数据 ,这里能很好的作为Dubbo服务的注册中心。
目标:封装好复杂易出错的关键服务,将简单易用的接口和性能高效、功能稳定的系统提供给用户
5. RPC
应用之间交互增多,将核心业务抽取出来,作为独立的服务,逐渐形成稳定的服务中心
RPC(remote procedure call protocol):远程过程调用
RPC是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的协议。
RPC协议假定某些传输协议的存在,如TCP或UDP,为通信程序之间携带信息数据。在OSI网络通信模型中,RPC跨越了传输层和应用层。
RPC采用客户机/服务器模式。请求程序就是一个客户机,而服务提供程序就是一个服务器。
a 首先,客户机调用进程发送一个有进程参数的调用信息到服务进程,然后等待应答信息。
b 在服务器端,进程保持睡眠状态直到调用信息到达为止。当一个调用信息到达,服务器获得进程参 数,计算结果,发送答复信息,然后等待下一个调用信息。
c 最后,客户端调用进程接收答复信息,获得进程结果,然后调用执行继续进行。
6. 搭建demo
步骤:
安装Zookeeper,启动;
创建MAVEN项目,构建Dubbo+Zookeeper+Spring实现的简单Demo;
安装Dubbo-admin,实现监控。
1、安装Zookeeper:http://blog.csdn.net/tlk20071/article/details/52028945
2、创建Maven项目,定义接口:
public interface DemoService { List<String> getPermissions(Long id); }
3、创建Maven项目(provider)
在服务方实现接口:
/** * 服务提供方:提供接口的实现类 */ public class DemoServiceImpl implements DemoService { public List<String> getPermissions(Long id) { List<String> permissions = new ArrayList<String>(); permissions.add(String.format("Permission_%d", id - 1)); permissions.add(String.format("Permission_%d", id)); permissions.add(String.format("Permission_%d", id + 1)); return permissions; } }
加载:
/** * 服务提供方 */ public class Provider { public static void main(String[] args) throws IOException { //从类路径下加载配置文件 ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("provider.xml"); System.out.println(context.getDisplayName() + ": here"); context.start(); System.out.println("服务提供方已经启动,开始提供服务..."); //按任意键退出 System.in.read(); } }
Spring配置声明暴露服务(provider.xml)
<?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-admin 或 dubbo-monitor 会显示这个名字,方便辨识--> <dubbo:application name="demotest-provider" owner="programmer" organization="dubbox"/> <!--使用 zookeeper 注册中心暴露服务,注意要先开启 zookeeper--> <dubbo:registry address="zookeeper://localhost:2181"/> <!-- 用dubbo协议在20880端口暴露服务 --> <dubbo:protocol name="dubbo" port="20880" /> <!--使用 dubbo 协议实现定义好的 api.DemoService 接口--> <dubbo:service interface="com.alibaba.dubbo.demo.DemoService" ref="demoService" protocol="dubbo" /> <!--具体实现该接口的 bean--> <bean id="demoService" class="com.alibaba.dubbo.demo.impl.DemoServiceImpl"/> </beans>
4、创建Maven项目(consumer)
配置spring,引用服务:
<?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:application name="demotest-consumer" owner="programmer" organization="dubbox"/> <!--向 zookeeper 订阅 provider 的地址,由 zookeeper 定时推送--> <dubbo:registry address="zookeeper://localhost:2181"/> <!--使用 dubbo 协议调用定义好的 api.DemoService接口--> <dubbo:reference id="demoService" interface="com.alibaba.dubbo.demo.DemoService" timeout="2000"/> <dubbo:consumer timeout="3000"/> <!--<dubbo:reference id="demoService" interface="com.alibaba.dubbo.demo.DemoService" check="false"/>--> </beans>
加载:
public class Consumer { public static void main(String[] args) { //测试常规服务 ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("consumer.xml"); context.start(); System.out.println("consumer1 start"); //获取远程服务 //DemoService demoService = (DemoService) context.getBean("demoService"); DemoService demoService = context.getBean(DemoService.class); System.out.println("consumer1:"); System.out.println(demoService.getPermissions(1L)); } }