基于Zookeeper实现客户端动态监听服务器上下线
一、在具体实现之前,先来了解一下Zookeeper的监听器的原理:

图中Main()线程作为客户端,当在主线程中创建Zookeeper客户端时,会默认创建两个子线程:Listener和connect,connect线程负责将某一操作对应的的监听事件发送给Zookeeper服务集群。Zookeeper收到监听事件后会在该操作对应的监听器列表中注册该事件。
比如图中的获取节点“/”的子节点getChildren这一事件,并设置了true,表示监听此事件。那么Zookeeper就会在监听器列表中注册该事件。一旦“/”节点的子节点发生变化,会导致getChildren发生变化,Zookeeper就会通知客户端的Listener线程,Listener就会去调用process方法对“/”的变化做出应对处理。“/”的变化可能是客户端不能控制的,但是为了适应这种变化,客户端在收到服务器的通知后可根据自身情况做出应对。
二、这样说可能比较抽象,我们用一个案例来说明:
public class ZkDemo {
private String connect = "hadoop101:2181";
private int timeout = 2000;
private ZooKeeper zooKeeper = null;
//1、获取Zookeeper客户端,用于连接Zookeeper集群,其功能类似于Linux中启动./zkCli.sh
@Before
public void getClient() throws Exception{
zooKeeper = new ZooKeeper(connect, timeout, new Watcher() {
@Override
public void process(WatchedEvent event) {
System.out.println(event.getPath() + "已被修改,是否确定?");
try {
zooKeeper.getChildren("/lsj",true);
} catch (KeeperException e) {
e.printStackTrace();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
}
/**
* 查看子节点
*/
@Test
public void testList() throws Exception {
List<String> children = zooKeeper.getChildren("/lsj", true);
for (String str: children) {
System.out.println(str);
}
Thread.sleep(Long.MAX_VALUE);
}
}
附:Thread.sleep()的设置是为了让testList方法从某种意义上更像是客户端,客户端时保持连接的,所以必须让testList方法处在执行的过程中。
默认的监听次数是一次,但是在getClient方法中使了一点小技巧使得监听的次数好像变得无限多了,只是好像稍后再说。只需记住:这段代码启动后对Zookeeper来说也是一个客户端,设为客户端1。在zooKeeper.getChildren("/lsj", true)设置了监听此事件,当/lsj节点下的子节点发生变化时getChildren方法结果会发生变化,会触发监听器Watcher(就是那个内部类)执行process方法,在process方法中执行了应对变化的处理后再次调用testList方法,这才使得监听好像变的无限多了。
下图所示:在Linux中启动的客户端(设为客户端2)每一次对/lsj节点增加节点都会视为getChildren结果的变化,故而Zookeeper会通知客户端1这种变化,进而触发监听器执行process方法。


三、下面进入Zookeeper的正题:如何利用Zookeeper监听服务器集群的变化,并在服务器集群变化时通知客户端呢?
原理和上面一样一样的,只需将客户端1想象成下文的客户端群体,把客户端2想想成服务器群体。如果没有Zookeeper这个角色,让客户端直接和服务器接触,当客户端请求的一台服务器正好宕机时,客户端将无法获取资源,但又不知道这是服务器宕机所造成的问题,也无法改变请求到另一台正常运行的服务器,那么这个问题如何解决呢?
思路是这样的,每一台服务器上线时都会在Zookeeper上的/ServerCluster的节点下创建一个标识本机的子节点(临时的 -e )。当,某一台服务器宕机时,那么其在/ServerCluster下创建的节点就会随之消失。我们让客户端获取服务器信息时,监听/ServerCluster的子节点变化,那么当某一台服务器宕机时(临时节点随之消失),Zookeeper会通知客户端/ServerCluster的变化,客户端也就知道了具体的哪一台服务器当机,哪一台服务器正常运行可以访问了。
在这个过程中,最重要的一点:服务器和客户端对于Zookeeper集群来说,都是客户端。

具体实现:
public class Server {
private static String connect = "hadoop101:2181";
private static int timeout = 2000;
private static ZooKeeper zooKeeper = null;
private static String parentPath = "/ServerCluster";
public static void main(String[] args) throws Exception {
//1、获取一个Zookeeper的客户端对象,用于服务器向Zookeeper集群注册自己。
getClient();
//2、把本服务器的主机名注册到Zookeeper中的特定节点中
registServer(args[0]);
//3、服务器本身的业务逻辑
getBusiness(args[0]);
}
private static void getBusiness(String hostname) throws InterruptedException {
System.out.println(hostname + " is working...");
Thread.sleep(Long.MAX_VALUE);
}
private static void registServer(String hostname) throws KeeperException, InterruptedException {
//创建临时节点
String path = zooKeeper.create(parentPath + "/server",
hostname.getBytes(),
ZooDefs.Ids.OPEN_ACL_UNSAFE,
CreateMode.EPHEMERAL_SEQUENTIAL);
System.out.println(hostname + " is online...");
}
private static void getClient() throws Exception {
zooKeeper = new ZooKeeper(connect, timeout, new Watcher() {
@Override
public void process(WatchedEvent event) {
System.out.println(event.getType() + "---" + event.getPath());
}
});
}
}
public class Client {
private static String connect = "hadoop101:2181";
private static int timeout = 2000;
private static ZooKeeper zooKeeper = null;
private static String parentPath = "/ServerCluster";
public static void main(String[] args) throws Exception {
//1、获取一个Zookeeper的客户端对象,用于服务器向Zookeeper集群注册自己。
getClient();
//2、获取服务器列表(主机名),并监听
getServers();
//3、客户端的业务逻辑
getBusiness();
}
private static void getBusiness() throws InterruptedException {
System.out.println("Client is working...");
Thread.sleep(Long.MAX_VALUE);
}
private static void getServers() throws KeeperException, InterruptedException {
//向Zookeeper给getChildren方法注册监听,一旦parentPath节点发生变化,就会通知监听器触发process方法
List<String> children = zooKeeper.getChildren(parentPath, true);
//此集合用于保存服务器主机名
ArrayList<String> hosts = new ArrayList();
for (String child: children) {
byte[] data = zooKeeper.getData(parentPath + "/" + child, false, null);
hosts.add(new String(data));
}
System.out.println(hosts);
}
private static void getClient() throws Exception {
zooKeeper = new ZooKeeper(connect, timeout, new Watcher() {
@Override
public void process(WatchedEvent event) {
System.out.println(event.getType() + "---" + event.getPath());
//执行到本方法就说明parentPath已经被修改了,即服务器列表发生变化,需要重新获取。
try {
getServers();
} catch (Exception e) {
e.printStackTrace();
}
}
});
}
}
通过配置参数开启三个服务器hadoop101,开启一个Client:

关闭hadoop101:

Client观察到这种变化,打印新的服务器列表:


当然,如果在Linux中加入一个“服务器”,Client也可以监听到:

基于Zookeeper实现客户端动态监听服务器上下线的更多相关文章
- oracle静态与动态监听
在运行lsnrctl命令的status时,常会看到如下返回值: 服务“test”包含1个例程. 例程"mydata",状态 UNKOWN,包含此服务的一个处理程序... 服务 ...
- 详解vuex结合localstorage动态监听storage的变化
这篇文章主要介绍了详解vuex结合localstorage动态监听storage的变化,小编觉得挺不错的,现在分享给大家,也给大家做个参考.一起跟随小编过来看看吧 需求:不同组件间共用同一数据,当一个 ...
- js动态监听dom变化
原生js 动态监听dom变化,根据不同的类型绑定不同的处理逻辑 // Firefox和Chrome早期版本中带有前缀 var MutationObserver = window.MutationO ...
- ORACLE动态监听
动态监听的原理 pmon在数据库启动到mount或open时,动态从参数文件中读取service_names值.service_names可以为多值(可以有64个,其中包括两个系统的). servi ...
- Linux下安装oracle遇到启动监听服务器启动失败
1.发现监听服务器没有启动,则 lsntctl start 启动监听服务器: 2.发现TNS-12555问题: 3.查找TNS-12555错误,找到一个满意的答案: chmod 777 /var/ ...
- Jmeter无法监听服务器4444端口
阿里云服务器开放了4444端口 jmeter还是无法监听: 解决方法: 阿里云安全组添加端口5555 服务器中启动监听插件使用5555端口,使用命令:java -jar ./CMDRunner.jar ...
- SpringMVC 监听文件上传进度
Spring MVC 监听文件上传进度 具体实现分三个步骤: 接管CommonsMultipartResolver,重写针对文件上传的请求. 在第一步中写入监听,以获取上传进度. 修改上传部分的配置文 ...
- [Swift通天遁地]三、手势与图表-(1)监听屏幕上触摸事件的各种状态
★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★➤微信公众号:山青咏芝(shanqingyongzhi)➤博客园地址:山青咏芝(https://www.cnblogs. ...
- zookeeper实现动态感知服务器上下线
在实际的生产环境中我们一般都是集群环境部署的,同一个程序我们会部署在相同的几台服务器中,这时我们可以通过负载均衡服务器去调度,但是我们并不能很快速的获知哪台服务器挂掉了,这时我们就可以使用zook ...
随机推荐
- C# 开发 Windows 服务 使用Log4net 组件 不能生成日志文件
使用VS2012开发Windows服务,需要使用Log4net日志组件记录业务情况,但是始终生成不了日志文件. /// <summary> /// 入口方法 /// </summar ...
- tmux 操作简版
操作session: 操作window: 操作pane: 原文
- python pip安装模块报错 "Can't connect to HTTPS URL because the SSL module is not available."
在升级python版本为3.6之后,pip安装模块报错. 报错信息如图: 原因是系统自带的openssl版本与python3的版本不匹配,所以这里只要升级openssl版本就可以解决问题. yum - ...
- 关于Python获取图片文件二进制数据的问题(获取为空)
在搭建fastdfs文件系统的时候遇到了点问题,在测试上传文件数据流的时候,需要Python来获取本地文件的二进制流 from fdfs_client.client import Fdfs_clien ...
- 程序员要注意!现在是RSS复兴的时候了
一般来说,现代网络不乏恐怖,从无所不在的网络黑客到所有信息平台,再到各大平台的评论系统.不幸的是,我们建立的这个互联网并没有什么灵丹妙药.但任何人都厌倦了黑箱算法,控制你在网上看到的东西,一直存在但始 ...
- spring boot 集成 websocket 实现消息主动推送
spring boot 集成 websocket 实现消息主动 前言 http协议是无状态协议,每次请求都不知道前面发生了什么,而且只可以由浏览器端请求服务器端,而不能由服务器去主动通知浏览器端,是单 ...
- JVM内存分配调优
Reference: https://time.geekbang.org/column/article/108139 参考指标 GC频率:⾼频的FullGC会给系统带来⾮常⼤的性能消耗,虽然Minor ...
- CMD批处理查看当前路径
1.直接在CMD窗口查看 echo %cd% 2.建立批处理文件 @echo offecho 当前盘符:%~d0echo 当前盘符和路径:%~dp0echo 当前批处理全路径:%~f0echo 当前盘 ...
- 配置中心git版本示例
1.运行环境 开发工具:intellij idea JDK版本:1.8 项目管理工具:Maven 4.0.0 2.GITHUB地址 https://github.com/nbfujx/springCl ...
- vim安装bundle和使用
一.准备工作 安装Git(因为下面我们选择的插件管理器需要使用到它)安装其他插件前首先需要选择一个Vim插件管理器,我这里选择的是Vundle,Vundle的工作过程中需要通过Git自动从远程创库同步 ...