不久前,我们讨论过Nginx+tomcat组成的集群,这已经是非常灵活的集群技术,但是当我们的系统遇到更大的瓶颈,全部应用的单点服务器已经不能满足我们的需求,这时,我们要考虑另外一种,我们熟悉的内容,就是分布式,而当下流行的Dubbo框架,不容我们忽视,这里,咱们一起来探讨一下这个框架的使用。
一,背景
以前我们需要远程调用他人的接口,我们是这么做的:
我们遇到的问题:
(1) 当服务越来越多时,服务URL配置管理变得非常困难,F5硬件负载均衡器的单点压力也越来越大。
此时需要一个服务注册中心,动态的注册和发现服务,使服务的位置透明。
并通过在消费方获取服务提供方地址列表,实现软负载均衡和Failover,降低对F5硬件负载均衡器的依赖,也能减少部分成本。
(2) 当进一步发展,服务间依赖关系变得错踪复杂,甚至分不清哪个应用要在哪个应用之前启动,架构师都不能完整的描述应用的架构关系。
这时,需要自动画出应用间的依赖关系图,以帮助架构师理清理关系。
(3) 接着,服务的调用量越来越大,服务的容量问题就暴露出来,这个服务需要多少机器支撑?什么时候该加机器?
为了解决这些问题,第一步,要将服务现在每天的调用量,响应时间,都统计出来,作为容量规划的参考指标。
其次,要可以动态调整权重,在线上,将某台机器的权重一直加大,并在加大的过程中记录响应时间的变化,直到响应时间到达阀值,记录此时的访问量,再以此访问量乘以机器数反推总容量。
为解决这些问题,Dubbo为我们做了什么呢:
负载均衡:
这就是所谓的软负载均衡!
现在让我们一起来接触下这个优秀的框架:
简介
架构如图:
节点角色说明:
Provider: 暴露服务的服务提供方。
Consumer: 调用远程服务的服务消费方。
Registry: 服务注册与发现的注册中心。
Monitor: 统计服务的调用次调和调用时间的监控中心。
Container: 服务运行容器。
调用关系说明:
0. 服务容器负责启动,加载,运行服务提供者。
1. 服务提供者在启动时,向注册中心注册自己提供的服务。
2. 服务消费者在启动时,向注册中心订阅自己所需的服务。
3. 注册中心返回服务提供者地址列表给消费者,如果有变更,注册中心将基于长连接推送变更数据给消费者。
4. 服务消费者,从提供者地址列表中,基于软负载均衡算法,选一台提供者进行调用,如果调用失败,再选另一台调用。
5. 服务消费者和提供者,在内存中累计调用次数和调用时间,定时每分钟发送一次统计数据到监控中心。
Dubbo提供了很多协议,Dubbo协议、RMI协议、Hessian协议,我们查看Dubbo源代码,有各种协议的实现,如图所示:
我们之前没用Dubbo之前时,大部分都使用Hessian来使用我们服务的暴露和调用,利用HessianProxyFactory调用远程接口。
上面是参考了Dubbo官方网介绍,接下来我们来介绍SpringMVC、Dubbo、Zookeeper整合使用。
第三:Dubbo与Zookeeper、SpringMVC整合使用
第一步:在Linux上安装Zookeeper
Zookeeper作为Dubbo服务的注册中心,Dubbo原先基于数据库的注册中心,没采用Zookeeper,Zookeeper一个分布式的服务框架,是树型的目录服务的数据存储,能做到集群管理数据 ,这里能很好的作为Dubbo服务的注册中心,Dubbo能与Zookeeper做到集群部署,当提供者出现断电等异常停机时,Zookeeper注册中心能自动删除提供者信息,当提供者重启时,能自动恢复注册数据,以及订阅请求。我们先在linux上安装Zookeeper,我们安装最简单的单点,集群比较麻烦。
先需要安装JdK,从Oracle的Java网站下载,安装很简单,就不再详述。
单机模式
单机安装非常简单,只要获取到 Zookeeper 的压缩包并解压到某个目录如:C:zookeeper-3.4.5下,Zookeeper 的启动脚本在 bin 目录下,Windows 下的启动脚本是 zkServer.cmd。
在你执行启动脚本之前,还有几个基本的配置项需要配置一下,Zookeeper 的配置文件在 conf 目录下,这个目录下有 zoo_sample.cfg 和 log4j.properties,你需要做的就是将 zoo_sample.cfg 改名为 zoo.cfg,因为 Zookeeper 在启动时会找这个文件作为默认配置文件。下面详细介绍一下,这个配置文件中各个配置项的意义。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
|
<span style= "font-size:18px;" ># The number of milliseconds of each tick
tickTime= 2000
# The number of ticks that the initial # synchronization phase can take initLimit= 10
# The number of ticks that can pass between # sending a request and getting an acknowledgement syncLimit= 5
# the directory where the snapshot is stored. # do not use /tmp for storage, /tmp here is just
# example sakes. dataDir=C:\zookeeper- 3.4 . 5 \data
dataLogDir=C:\zookeeper- 3.4 . 5 \log
# the port at which the clients will connect clientPort= 2181
# # Be sure to read the maintenance section of the # administrator guide before turning on autopurge. # # https: //zookeeper.apache.org/doc/current/zookeeperAdmin.html#sc_maintenance
# # The number of snapshots to retain in dataDir #autopurge.snapRetainCount= 3
# Purge task interval in hours # Set to 0 to disable auto purge feature
#autopurge.purgeInterval= 1 </span>
|
tickTime:这个时间是作为 Zookeeper 服务器之间或客户端与服务器之间维持心跳的时间间隔,也就是每个 tickTime 时间就会发送一个心跳。dataDir:顾名思义就是 Zookeeper 保存数据的目录,默认情况下,Zookeeper 将写数据的日志文件也保存在这个目录里。dataLogDir:顾名思义就是 Zookeeper 保存日志文件的目录clientPort:这个端口就是客户端连接 Zookeeper 服务器的端口,Zookeeper 会监听这个端口,接受客户端的访问请求。
当这些配置项配置好后,你现在就可以启动 Zookeeper 了,启动后要检查 Zookeeper 是否已经在服务,可以通过 netstat – ano 命令查看是否有你配置的 clientPort 端口号在监听服务。
第二步:配置dubbo-admin的管理页面,方便我们管理页面
(1)下载dubbo-admin-2.4.1.war包,在windows的tomcat部署,先把dubbo-admin-2.4.1放在tomcat的webapps/ROOT下,然后进行解压
(2)然后到webapps/ROOT/WEB-INF下,有一个dubbo.properties文件,里面指向Zookeeper ,使用的是Zookeeper 的注册中心,如图所示:
1
2
3
|
<span style= "font-size:18px;" >dubbo.registry.address=zookeeper: //127.0.0.1:2181
dubbo.admin.root.password=root dubbo.admin.guest.password=guest</span> |
(3)然后启动tomcat服务,用户名和密码:root,并访问服务,显示登陆页面,说明dubbo-admin部署成功,如图所示:
第三步:SpringMVC与Dubbo的整合,这边使用的Maven的管理项目
第一:我们先开发服务注册的,就是提供服务,项目结构如图所示:
(1)test-maven-api项目加入了一个服务接口,代码如下:
1
2
3
|
public interface TestRegistryService {
public String hello(String name);
} |
(2)test-maven-console在pom.xml加入Dubbo和Zookeeper的jar包、引用test-maven-api的jar包,代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
<span style= "font-size:18px;" ><dependency>
<groupid>cn.test</groupid>
test-maven-api</artifactid>
<version> 0.0 . 1 -SNAPSHOT</version>
</dependency>
<dependency>
<groupid>com.alibaba</groupid>
dubbo</artifactid>
<version> 2.5 . 3 </version>
</dependency>
<dependency>
<groupid>org.apache.zookeeper</groupid>
zookeeper</artifactid>
<version> 3.4 . 6 </version>
</dependency>
<dependency>
<groupid>com.github.sgroschupf</groupid>
zkclient</artifactid>
<version> 0.1 </version>
</dependency></span> |
(3)test-maven-console实现具体的服务,代码如下:
1
2
3
4
5
6
|
@Service (testRegistryService)
public class TestRegistryServiceImpl implements TestRegistryService {
public String hello(String name) {
return hello+name;
}
} |
(4)我们服务以及实现好了,这时要暴露服务,代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
<!--?xml version= 1.0 encoding=UTF- 8 ?-->
<beans span= "" style= "color:#cc0000;" xmlns= "https://www.springframework.org/schema/beans" xmlns:jee= "https://www.springframework.org/schema/jee" xmlns:tx= "https://www.springframework.org/schema/tx" xmlns:xsi= "https://www.w3.org/2001/XMLSchema-instance" >xmlns:dubbo=https://code.alibabatech.com/schema/dubbo
xmlns:context=https: //www.springframework.org/schema/context
xsi:schemaLocation=https: //www.springframework.org/schema/beans https://www.springframework.org/schema/beans/spring-beans-3.1.xsd
https: //www.springframework.org/schema/tx https://www.springframework.org/schema/tx/spring-tx-3.1.xsd
https: //www.springframework.org/schema/jee https://www.springframework.org/schema/jee/spring-jee-3.1.xsd
<span style= "color:#990000;" >https: //code.alibabatech.com/schema/dubbo https://code.alibabatech.com/schema/dubbo/dubbo.xsd</span>
https: //www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context-3.1.xsd
default -lazy-init= false >
<!-- 提供方应用名称信息,这个相当于起一个名字,我们dubbo管理页面比较清晰是哪个应用暴露出来的 -->
<dubbo:application name= "dubbo_provider" ></dubbo:application>
<!-- 使用zookeeper注册中心暴露服务地址 -->
<dubbo:registry address= "zookeeper://127.0.0.1:2181" check= "false" register= "" subscribe= "false" ></dubbo:registry>
<!-- 要暴露的服务接口 -->
<dubbo:service interface = "cn.test.dubbo.registry.service.TestRegistryService" ref= "testRegistryService" >
</dubbo:service></beans> |
说明:
dubbo:registry 标签一些属性的说明:
1)register是否向此注册中心注册服务,如果设为false,将只订阅,不注册。
2)check注册中心不存在时,是否报错。
3)subscribe是否向此注册中心订阅服务,如果设为false,将只注册,不订阅。
4)timeout注册中心请求超时时间(毫秒)。
5)address可以Zookeeper集群配置,地址可以多个以逗号隔开等。
dubbo:service标签的一些属性说明:
1)interface服务接口的路径
2)ref引用对应的实现类的Bean的ID
3)registry向指定注册中心注册,在多个注册中心时使用,值为的id属性,多个注册中心ID用逗号分隔,如果不想将该服务注册到任何registry,可将值设为N/A
4)register 默认true ,该协议的服务是否注册到注册中心。
(5)启动项目,然后我们在Dubbo管理页面上显示,已经暴露的服务,但显示还没有消费者,因为我们还没实现消费者服务,如图所示:
第二:我们在开发服务消费者,就是调用服务,我们在新建一个新的消费者项目:
(1)test-maven-server-console的pom.xml引入Dubbo和Zookeeper的jar包、test-maven-api的jar包,因为引入test-maven-api的jar包,我们在项目中调用像在本地调用一样。代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
<span style= "font-size:18px;" ><dependency>
<groupid>cn.test</groupid>
test-maven-api</artifactid>
<version> 0.0 . 1 -SNAPSHOT</version>
</dependency>
<dependency>
<groupid>com.alibaba</groupid>
dubbo</artifactid>
<version> 2.5 . 3 </version>
</dependency>
<dependency>
<groupid>org.apache.zookeeper</groupid>
zookeeper</artifactid>
<version> 3.4 . 6 </version>
</dependency>
<dependency>
<groupid>com.github.sgroschupf</groupid>
zkclient</artifactid>
<version> 0.1 </version>
</dependency></span> |
(2)test-maven-server-console项目的具体实现,代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
@Controller public class IndexController {
@Autowired
private TestRegistryService testRegistryService;
@RequestMapping (/hello)
public String index(Model model){
String name=testRegistryService.hello(zz);
System.out.println(xx==+name);
return ;
}
} |
(3)我们要引用的地址,代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
<!--?xml version= 1.0 encoding=UTF- 8 ?-->
<beans span= "" style= " background: none !important; border: 0px !important; bottom: auto !important; float: none !important; height: auto !important; left: auto !important; line-height: 1.1em !important; margin: 0px !important; outline: 0px !important; overflow: visible !important; padding: 0px !important; position: static !important; right: auto !important; top: auto !important; vertical-align: baseline !important; width: auto !important; box-sizing: content-box !important; font-family: Consolas, "Bitstream Vera Sans Mono", "Courier New", Courier, monospace !important; font-size: 1em !important; min-height: auto !important;">xmlns= "https://www.springframework.org/schema/beans" xmlns:jee= "https://www.springframework.org/schema/jee" xmlns:tx= "https://www.springframework.org/schema/tx" xmlns:xsi= "https://www.w3.org/2001/XMLSchema-instance" ><span style= "color:#990000;" >xmlns:dubbo=https://code.alibabatech.com/schema/dubbo</span>
xmlns:context=https: //www.springframework.org/schema/context
xsi:schemaLocation=https: //www.springframework.org/schema/beans https://www.springframework.org/schema/beans/spring-beans-3.1.xsd
https: //www.springframework.org/schema/tx https://www.springframework.org/schema/tx/spring-tx-3.1.xsd
https: //www.springframework.org/schema/jee https://www.springframework.org/schema/jee/spring-jee-3.1.xsd
<span style= "color:#990000;" >https: //code.alibabatech.com/schema/dubbo https://code.alibabatech.com/schema/dubbo/dubbo.xsd</span>
https: //www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context-3.1.xsd
default -lazy-init= false >
<dubbo:application name= "dubbo_consumer" ></dubbo:application>
<!-- 使用zookeeper注册中心暴露服务地址 -->
<!-- 要引用的服务 -->
<dubbo:reference id= "testRegistryService" interface = "cn.test.dubbo.registry.service.TestRegistryService" ></dubbo:reference>
</beans> |
说明:
dubbo:reference 的一些属性的说明:
1)interface调用的服务接口
2)check 启动时检查提供者是否存在,true报错,false忽略
3)registry 从指定注册中心注册获取服务列表,在多个注册中心时使用,值为的id属性,多个注册中心ID用逗号分隔
4)loadbalance 负载均衡策略,可选值:random,roundrobin,leastactive,分别表示:随机,轮循,最少活跃调用
(4)项目启动,Dubbo管理页面,能看到消费者,如图所示:
(5)然后访问消费者项目,Controller层能像调用本地一样调用服务的具体实现,如图所示:
总结:
经过一系列的操作之后,我们感觉,的确很简单,dubbo给我们封装了很多操作,让我们不需要过多考虑具体的实现细节,配置化生成咱们的应用,这样的思想,现在在IT行业正在盛行!
Dubbo+zookeeper构建高可用分布式集群(一)-单机部署的更多相关文章
-
Dubbo+zookeeper构建高可用分布式集群(二)-集群部署
在Dubbo+zookeeper构建高可用分布式集群(一)-单机部署中我们讲了如何单机部署.但没有将如何配置微服务.下面分别介绍单机与集群微服务如何配置注册中心. Zookeeper单机配置:方式一. ...
-
Java集群优化——dubbo+zookeeper构建高可用分布式集群
不久前,我们讨论过Nginx+tomcat组成的集群,这已经是非常灵活的集群技术,但是当我们的系统遇到更大的瓶颈,全部应用的单点服务器已经不能满足我们的需求,这时,我们要考虑另外一种,我们熟悉的内容, ...
-
.net core下简单构建高可用服务集群
一说到集群服务相信对普通开发者来说肯定想到很复杂的事情,如zeekeeper ,反向代理服务网关等一系列的搭建和配置等等:总得来说需要有一定经验和规划的团队才能应用起来.在这文章里你能看到在.net ...
-
redis高可用分布式集群
一,高可用 高可用(High Availability),是当一台服务器停止服务后,对于业务及用户毫无影响. 停止服务的原因可能由于网卡.路由器.机房.CPU负载过高.内存溢出.自然灾害等不可预期的原 ...
-
ElasticSearch 高可用分布式集群搭建,与PHP多线程测试
方案: 使用HAproxy:当其中一台ElasticSearch Master宕掉时,ElasticSearch集群会自动将运行正常的节点提升为Master,但HAproxy不会将失败的请求重新分发到 ...
-
Redis 高可用分布式集群
一,高可用 高可用(High Availability),是当一台服务器停止服务后,对于业务及用户毫无影响. 停止服务的原因可能由于网卡.路由器.机房.CPU负载过高.内存溢出.自然灾害等不可预期的原 ...
-
redis详解(四)-- 高可用分布式集群
一,高可用 高可用(High Availability),是当一台服务器停止服务后,对于业务及用户毫无影响. 停止服务的原因可能由于网卡.路由器.机房.CPU负载过高.内存溢出.自然灾害等不可预期的原 ...
-
一张图讲解最少机器搭建FastDFS高可用分布式集群安装说明
很幸运参与零售云快消平台的公有云搭建及孵化项目.零售云快消平台源于零售云家电3C平台私有项目,是与公司业务强耦合的.为了适用于全场景全品类平台,集团要求项目平台化,我们抢先并承担了此任务.并由我来主 ...
-
hadoop3.1.1 HA高可用分布式集群安装部署
1.环境介绍 涉及到软件下载地址:https://pan.baidu.com/s/1hpcXUSJe85EsU9ara48MsQ 服务器:CentOS 6.8 其中:2 台 namenode.3 台 ...
随机推荐
-
输入一个整型数组,数据元素有正数也有负数,求元素组合成连续子数组之和最大的子数组,要求时间复杂度为O(n)。
如果不考虑时间复杂度,我们可以枚举出所有子数组并求出他们的和.不过非常遗憾的是,由于长度为n的数组有O(n2)个子数组(即:n + n-1 + ... + 1=n(n+1)/2):而且求一个长度为n的 ...
-
bat写的自动部署脚本
windows7的机器上重启服务需要关闭UAC ::编译部署项目 echo off echo 1. GatewayAdaptor echo 2. LogicService echo 3. Messag ...
-
android 沉浸通知栏
IOS的沉浸式通知栏很高大上,通知栏和app统一颜色或样式,很美观.android上面也早就人实现这种效果了. 我在这边也写一个实现通知栏沉浸式的方法,目前只实现了相同颜色. 先要改布局文件xml & ...
-
YT新人之巅峰大决战04
Problem Description Eddy's interest is very extensive, recently he is interested in prime number. Ed ...
-
Java试题
1.不使用循环,等比数列输出整型 n.2n.4n.8n--当大于max时,反向输出8n.4n.2n.n. 例如 n=10,max=100. 输出: 10 20 40 80 80 40 20 10 解题 ...
-
ubuntu16.04使用Qt开发ROS
本文介绍一种Qt下进行ROS开发的完美方案,使用的是ros-industrial的Levi-Armstrong在2015年12月开发的一个Qt插件ros_qtc_plugin,这个插件使得Qt“新建项 ...
-
rsync同步工具的配置与使用
一.什么是rsync?rsync是一款开源的,快速的,多功能的,可实现全量及增量的本地或远程数据同步备份的优秀工具. rsync官网 http://rsync.samba.org/ 二.rsync的工 ...
-
【LeetCode题解】3_无重复字符的最长子串(Longest-Substring-Without-Repeating-Characters)
目录 描述 解法一:暴力枚举法(Time Limit Exceeded) 思路 Java 实现 Python 实现 复杂度分析 解法二:滑动窗口(双指针) 思路 Java 实现 Python 实现 复 ...
-
curl传输文件实例
curl -H "Authorization:Bearer 5d719398-4230-44c7-b88b-f280b6a8d070" -H "Accept: appli ...
-
bzoj 1520 [POI2006]Szk-Schools 费用流
[POI2006]Szk-Schools Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 743 Solved: 381[Submit][Status][ ...