基于zookeeper的主备切换方法

继承CZookeeperHelper即可快速实现主备切换: https://github.com/eyjian/mooon/blob/master/mooon/include/mooon/net/zookeeper_helper.h zookeeper的ZOO_EPHEMERAL节点(如果ZOO_EPHEMERAL满足不了需求,可以考虑和ZOO_SEQUENCE结合使用),在会话关闭或过期时,会自动删除,利用这一特性可以实现两个或多节点间的主备切换。 实现方法: 1)在进程启动时调用zookeeper_init()初始化: bool X::init_zookeeper() {     // 第一次调用时_clientid总是为NULL,     // 状态为ZOO_EXPIRED_SESSION_STATE时,需要重新调用zookeeper_init,     // 这个时候可传入的_clientid为前一次zookeeper_init()产生的_clientid     // 请注意zookeeper_init()是一个异步调用,返回非NULL并不表示会话建立成功,     // 只有当zk_watcher中的type为ZOO_SESSION_EVENT和state为ZOO_CONNECTED_STATE时,     // 才真正表示会话建立成功。     _zhandle = zookeeper_init(zk_hosts, zk_watcher, 5000, _clientid, this, 0);     if (NULL == _zhandle)     {         MYLOG_ERROR("init zookeeper failed: %s\n", zerror(errno));         return false;     }     MYLOG_INFO("init zookeeper(%s) successfully\n", zk_hosts);     return true; } 2)进入工作之前,先尝试切换成主,只有成功切换成主后才进入work bool X::run() {     while (true)     {         int num_items = 0;         // 备机最简单的方法是每隔一定时间,如1秒就尝试转成master,         // 如果不使用轮询,则可以采用监视_zk_path的方式         mooon::sys::CUtils::millisleep(1000);         // 如果不是master,则尝试转成master,如果转成不成功则继续下一次尝试         if (!is_master() && !change_to_master())             continue;         do_work();     } } bool X::is_master() const {     return _is_master; } bool X::change_to_master() {     static uint64_t log_counter = 0; // 打log计数器,备状态时的日志输出     // ZOO_EPHEMERAL|ZOO_SEQUENCE     // _myip为本地IP地址,可以通过它来判断当前谁是master     // _zk_path值示例:/master/test,注意需要先保证/master已存在     int errcode = zoo_create(_zhandle, _zk_path.c_str(), _myip.c_str(), _myip.size()+1, &ZOO_OPEN_ACL_UNSAFE, ZOO_EPHEMERAL, NULL, 0);     // (-4)connection loss,比如为zookeeper_init()指定了无效的hosts(一个有效的host也没有)     if (errcode != ZOK)     {         _is_master = false;         // 减少为备状态时的日志输出         if (0 == log_counter++ % 600)         {             MYLOG_DEBUG("become master[%s] failed: (%d)%s\n", _zk_path.c_str(), errcode, zerror(errcode));         }         return false;     }     else     {         _is_master = true;         log_counter = 0;         MYLOG_INFO("becase master[%s]\n", _zk_path.c_str());         // sleep一下,以便让原master正在进行的完成         mooon::sys::CUtils::millisleep(2000);         return true;     } } 3)当zookeeper会话成功建立或过期时均会触发zk_watcher,可通过type和state来区分 void zk_watcher(zhandle_t *zh, int type, int state, const char *path, void *context) {     X* x = static_cast(context);     MYLOG_DEBUG("zh=%p, type=%d, state=%d, context=%p, path=%s\n", zh, type, state, context, path);     // zookeeper_init成功时type为ZOO_SESSION_EVENT,state为ZOO_CONNECTED_STATE     if ((ZOO_SESSION_EVENT == type) && (ZOO_CONNECTED_STATE == state))     {         x->on_zookeeper_connected(path);     }     else if ((ZOO_SESSION_EVENT == type) && (ZOO_EXPIRED_SESSION_STATE == state))     {         // 需要重新调用zookeeper_init(),简单点可以退出当前进程重启         x->on_zookeeper_expired();     } } 附: zookeeper日志 默认情况下zookeeper日志是输出到stderr,但可以通过zoo_set_log_stream()来定向到自己的日志输出中,还可以使用zoo_set_debug_level()来控制zookeeper的日志级别。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏清晨我上码

第六节 netty前传-NIO Selector

可以使用单个线程来处理多个channel来节省资源。对于操作系统而言,线程之间切换是昂贵的,并且每个线程也占用操作系统中的一些资源(存储器)。 因此,使用的线程...

1142
来自专栏专注于主流技术和业务

SpringMVC源码阅读:ContextLoaderListener初始化过程

ContextLoaderListener监听器的作用就是启动web容器时,自动装配ApplicationContext的配置信息。它实现了ServletCon...

1904
来自专栏javathings

Java 的 NIO 是如何工作的?

在这个数据爆炸的时代,有大量的数据在系统中流动,一个应用系统的瓶颈往往都是 IO 瓶颈。传统的 javaIO 模型是 BIO,也就是同步阻塞 IO,数据在写入 ...

1251
来自专栏java 成神之路

NIO 之 Channel实现原理

45514
来自专栏xdecode

Netty与传统Server对比

前言 本文旨在介绍传统Socket服务端与NIO服务端的差异. 以餐厅服务员简单举例,每个客人对应一个请求. 传统Socket / OIO 1 public ...

2547
来自专栏社区的朋友们

TAF 必修课(二):Reactor多线程模型

最近看了很多文章和分享,非常受益, 实习所做项目主要用到了TAF,有必要对之前的学习做个梳理和总结,网络线程模型及请求接收过程,必修亦为基础、通用,故取其名。

8441
来自专栏JavaEdge

Tomcat架构解析之3 Connector NIOAcceptorPollerWorkerNioSelectorPool

2764
来自专栏芋道源码1024

【Netty 专栏】Netty 源码分析之 accept 过程

在Netty源码分析之NioEventLoop章节中,已经分析了NioEventLoop的工作机制,当有客户端connect请求,selector可以返回其对应...

1050
来自专栏架构师之旅

【Java SE】Java NIO系列教程(八) SocketChannel

英文:Jakob Jenkov 译文:ifeve - 郑玉婷 链接:http://ifeve.com/socket-channel/ Java NIO中的Soc...

2107
来自专栏chenssy

【死磕Netty】-----NIO基础详解

原文出处http://cmsblogs.com/ 『chenssy』 转载请注明原创出处,谢谢! Netty 是基于Java NIO 封装的网络通讯框架,只有充...

4446

扫码关注云+社区