通过zookeeper完成动态感知分布式服务器上下线的功能

通过zookeeper完成动态感知分布式服务器上下线的功能

[TOC]

业务描述

  • 某分布式系统中,主节点可以有多台,可以动态上下线,任意一台客户端都能实时感知到主节点服务器的上下线。当主节点下线的时候,客户端会收到通知,并更新目前在还在线的主节点host信息,可以防止客户端向已经挂掉的节点进行请求。

服务器端的实现

  • 要想完成上述功能,在我们可以想到通过zookeeper的短暂态的节点完成,在服务器启动的时候,我们连接到zookeeper并向其注册一个短暂态的节点,当服务器因为某种意外宕机的时候,这个节点也会被删除,这样客户端访问所有的注册节点的信息,就是仍然在正常工作的主节点。

  • 服务器端的代码如下:

    import java.io.IOException;
    
    import org.apache.zookeeper.CreateMode;
    import org.apache.zookeeper.KeeperException;
    import org.apache.zookeeper.WatchedEvent;
    import org.apache.zookeeper.Watcher;
    import org.apache.zookeeper.ZooDefs.Ids;
    import org.apache.zookeeper.ZooKeeper;
    
    public class ClusterServer {
      
          //zookeeper的ip与端口,这里是zookeeper是一个集群
      private String ZK_SERVER_LIST = "mini1:2181,mini2:2181,mini3:2181";
      //设置超时时间,当主节点2s后没反应就认该节点已经挂了
      private static final int sessionTimeout = 2000;
      //所有的注册节点信息当道/servers下面,方便管理
      private String SERVER_DIR = "/servers";
      
      private ZooKeeper zk;
    
      public void connect() throws IOException {
          
          zk = new ZooKeeper(ZK_SERVER_LIST, sessionTimeout, new Watcher() {
              
              //当zookeeper服务器集群有断线时会调用
              @Override
              public void process(WatchedEvent event) {
                  // TODO Auto-generated method stub
                  System.out.println(event.toString());
              }
          });
      }
    
      public void register(String hontname) throws IOException, KeeperException, InterruptedException {
          //创建瞬时态且带序号的节点,这样在节点下线后该节点就会被删除
          String result = zk.create(SERVER_DIR+"/server", hontname.getBytes(), Ids.OPEN_ACL_UNSAFE, CreateMode.EPHEMERAL_SEQUENTIAL);
          
          System.out.println("创建结果:"+result);
      }
      
      public void diy(){
          System.out.println("just do it");
          try {
              Thread.sleep(Integer.MAX_VALUE);
          } catch (InterruptedException e) {
              // TODO Auto-generated catch block
              e.printStackTrace();
          }
      }
      
      public static void main(String[] args) throws IOException, KeeperException, InterruptedException {
          ClusterServer server =new ClusterServer();
          server.connect();
          server.register(args[0]);
          server.diy();
      }
    
    }
    
  • 程序启动的时候就向服务器把当前主节点的host信息注册到zookeeper中。

  • 这里的hostname本来可以通过java api进行动态获取的,不过为了方便实验就省略了。

客户端的实现

  • 客户端所需要的工作是获取正常工作的主节点并且当主节点发生变化的时候可以收到信息,获取最新的正常工作的主节点。所以我们可以对servers下面的子节点信息进行监听,

  • 代码实现

    import java.io.IOException;
    import java.util.ArrayList;
    import java.util.List;
    
    import org.apache.zookeeper.KeeperException;
    import org.apache.zookeeper.WatchedEvent;
    import org.apache.zookeeper.Watcher;
    import org.apache.zookeeper.ZooKeeper;
    
    public class ClusterClient {
      private String ZK_SERVER_LIST = "mini1:2181,mini2:2181,mini3:2181";
    
      private static final int sessionTimeout = 2000;
    
      private String SERVER_DIR = "/servers";
    
      private ZooKeeper zk;
      
      private List<String> servers=new ArrayList<>();
      public void connect() throws IOException {
          zk = new ZooKeeper(ZK_SERVER_LIST, sessionTimeout, new Watcher() {
    
              @Override
              public void process(WatchedEvent event) {
                  System.out.println(event.toString());
                  try {
                          //收到通知后再次对服务器设置监听。
                      getServerList();
                      System.out.println(servers);
                  } catch (KeeperException | InterruptedException e) {
                          
                  }
              }
          });
      }
      
          //获取服务器列表并设置监听
      public void getServerList() throws KeeperException, InterruptedException{
          List<String> nodeList = zk.getChildren(SERVER_DIR,true);
          servers.clear();
          for(String node:nodeList){
              byte []data = zk.getData(SERVER_DIR+"/"+node, false, null);
              String hostName = new String(data);
              servers.add(hostName);
          }
      }
      
          //做自己爱做的事同时完成守护进程的功能
      public void diy(){
          new Thread(new Runnable() {
              
              @Override
              public void run() {
                  // TODO Auto-generated method stub
                  while (true){
                      
                  }
              }
          }).start();;
      }
      
      public static void main(String[] args) throws IOException, KeeperException, InterruptedException {
          ClusterClient client =new ClusterClient();
          client.connect();
          client.getServerList();
          client.diy();
      }
    }
    
    

测试

  • 我们可以在多个终端分别运行服务器程序,用来模拟多个主节点,直接关闭终端模仿主节点的宕机。

  • 实验结果如下(当我们动态增加节点的结果):

  • 所有需要的jar包在zookeeper的解压包里面都有。或者直接在我的github里面有代码以及jar包。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,590评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,808评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,151评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,779评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,773评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,656评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,022评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,678评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,038评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,756评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,411评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,005评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,973评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,053评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,495评论 2 343

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,596评论 18 139
  • 1 Zookeeper概述# ZooKeeper是一个为分布式应用所设计的分布的、开源的协调服务,它主要是用来解决...
    七寸知架构阅读 7,333评论 0 101
  • ZooKeeper介绍 Zookeeper 分布式服务框架是 Apache Hadoop 的一个子项目,它主要是用...
    Coselding阅读 11,376评论 1 37
  • 什么是中式传统婚礼?就是以礼规范婚,让婚礼者和观礼者,从中体会到爱情的珍贵、亲情的高尚,从而,自觉的建立好自己的家...
    永远的mark阅读 397评论 0 0
  • 《欢乐颂》我们看完了,樊胜美让我们觉得她既可怜又可笑,可怜她的家庭,可笑她的自尊!对于樊胜美的父母,在追剧...
    12345believe阅读 453评论 1 0