分布式Dubbo+Zookeeper+SpringBoot
分布式系统是若干个独立计算机的集合,这些计算机对用户来说就像单个相关系统
RPC
- RPC(Remote Procedure Call)远程过程调用,简单的理解是一个节点请求另一个节点提供的服务
- 本地过程调用:如果需要将本地student对象的age+1,可以实现一个addAge()方法,将student对象传入,对年龄进行更新之后返回即可,本地方法调用的函数体通过函数指针来指定。
- 远程过程调用:上述操作的过程中,如果addAge()这个方法在服务端,执行函数的函数体在远程机器上,如何告诉机器需要调用这个方法呢?
- 首先客户端需要告诉服务器,需要调用的函数,这里函数和进程ID存在一个映射,客户端远程调用时,需要查一下函数,找到对应的ID,然后执行函数的代码。
- 客户端需要把本地参数传给远程函数,本地调用的过程中,直接压栈即可,但是在远程调用过程中不再同一个内存里,无法直接传递函数的参数,因此需要客户端把参数转换成字节流,传给服务端,然后服务端将字节流转换成自身能读取的格式,是一个序列化和反序列化的过程。
3.数据准备好了之后,如何进行传输?网络传输层需要把调用的ID和序列化后的参数传给服务端,然后把计算好的结果序列化传给客户端,因此TCP层即可完成上述过程,gRPC中采用的是HTTP2协议。
- 通讯
- 序列化
- 数据传输需要转换
ZooKeeper
ZooKeeper是一种集中式服务,用于维护配置信息,命名,提供分布式同步和提供组服务。所有这些类型的服务都以分布式应用程序的某种形式使用。每次实施它们都需要做很多工作来修复不可避免的错误和竞争条件。由于难以实现这些类型的服务,应用程序最初通常会吝啬它们,这使得它们在变化的情况下变得脆弱并且难以管理。即使正确完成,这些服务的不同实现也会在部署应用程序时导致管理复杂性。
下载解压后,运行服务端,要是闪退就查看conf文件下是否有zoo.cfg文件
https://zookeeper.apache.org/releases.html
- create -e /user pwd 创建
Dubbo
是一个基于java实现的高性能RPC通信框架!
因为zookeeper只是一个黑框,我们无法看到是否存在了什么提供者或消费者,这时就要借助Dubbo-Admin管理平台来实时的查看,也可以通过这个平台来管理提者和消费者。
- 下载解压https://github.com/apache/dubbo-admin/tree/master
- 在解压缩文件路径下运行cmd指令
mvn package -Dmaven.skip.test=true
打包jar包(运行不了需要检查maven环境) - 开启zookeeper服务端前提下运行jar包
- 访问7001端口,默认账户密码都是root
Dubbo+Zookeeper
提供者和消费者导入依赖
<!-- https://mvnrepository.com/artifact/org.apache.dubbo/dubbo-spring-boot-starter -->
<dependency>
<groupId>org.apache.dubbo</groupId>
<artifactId>dubbo-spring-boot-starter</artifactId>
<version>2.7.8</version>
</dependency>
<!-- https://mvnrepository.com/artifact/com.github.sgroschupf/zkclient -->
<dependency>
<groupId>com.github.sgroschupf</groupId>
<artifactId>zkclient</artifactId>
<version>0.1</version>
</dependency>
<dependency>
<groupId>org.apache.curator</groupId>
<artifactId>curator-framework</artifactId>
<version>2.12.0</version>
</dependency>
<dependency>
<groupId>org.apache.curator</groupId>
<artifactId>curator-recipes</artifactId>
<version>2.12.0</version>
</dependency>
<dependency>
<groupId>org.apache.zookeeper</groupId>
<artifactId>zookeeper</artifactId>
<version>3.4.14</version>
<!--排除log4j,不然会冲突-->
<exclusions>
<exclusion>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-log4j12</artifactId>
</exclusion>
</exclusions>
</dependency>
步骤
-
提供者提供服务
-
导入依赖
-
配置文件注册中心,配置名字/地址/扫描的包
server.port=8081 #生产者注册服务 #服务应用名字(项目名) dubbo.application.name=provider-server #注册的中心地址 dubbo.registry.address=zookeeper://127.0.0.1:2181 #注册那些服务 dubbo.scan.base-packages=com.mofei.service
-
提供服务的类需要添加注解
@DubboService
@DubboService //把服务注册到Dubbo去 public class TicketServiceImpl implements TicketService { @Override public String getTicket() { return "Ticket"; } }
-
-
消费者消费服务
-
导入依赖
-
配置文件注册中心,名字和地址
server.port=8082 #消费者去那个地方拿服务 #暴露自己名字 dubbo.application.name=consumer-server #注册中心 dubbo.registry.address=zookeeper://127.0.0.1:2181
-
从远程导入服务
@Reference
,需要同路径同接口名@Service//放入容器中 public class userService { //想拿到生产者注册的服务,需要去注册中心拿 @Reference//不要倒错包org.apache.dubbo ,引用远程,可以定义相同路径接口名取 TicketService ticketService; public void buyTicket(){ System.out.println("在注册中心拿到=>"+ticketService.getTicket()); } }
-
测试时候需要开zookeeper启服务端
@Autowired
userService userService;
@Test
void contextLoads() {
userService.buyTicket();
}
by:感谢狂神老师免费的springboot教程