Zookeeper C++编程实战之主备切换
默认zookeeper日志输出到stderr,
可以调用zoo_set_log_stream(FILE*)设置输出到文件中
还可以调用zoo_set_debug_level(ZooLogLevel)控制日志级别!!!
类CZookeeperHelper提供基于zookeeper的主备切换接口和读取数据等接口:
https://github.com/eyjian/libmooon/blob/master/include/mooon/net/zookeeper_helper.h
使用示例:
class CMyApplication: public mooon::net::CZookeeperHelper
{
public:
CMyApplication(const char* data);
void stop() { _stop = true; }
void run();
void wait(); private:
void work(); private:
virtual void on_zookeeper_session_connected(const char* path);
virtual void on_zookeeper_session_connecting(const char* path);
virtual void on_zookeeper_session_expired(const char *path);
virtual void on_zookeeper_session_event(int state, const char *path);
virtual void on_zookeeper_event(int type, int state, const char *path); private:
volatile bool _stop;
std::string _master_path; 用来竞争master的zookeeper节点路径
std::string _master_data; 成功竞争为master时,写入_master_path的数据,主备应当提供不同的数据,以方便判断自己是否处于主状态
}; int main(int argc, char* argv[])
{
try
{
mooon::sys::g_logger = mooon::sys::create_safe_logger();
const std::string zk_nodes = "127.0.0.1:2181";
const int session_timeout_seconds = 1; CMyApplication myapp(argv[1]);
myapp.create_session(zk_nodes, session_timeout_seconds);
myapp.run();
myapp.wait(); return 0;
}
catch (mooon::sys::CSyscallException& ex)
{
fprintf(stderr, "%s\n", ex.str().c_str());
exit(1);
}
catch (mooon::utils::CException& ex)
{
fprintf(stderr, "%s\n", ex.str().c_str());
exit(1);
}
} CMyApplication::CMyApplication(const char* data)
: _stop(false)
{
_master_path = "/tmp/a";
if (data != NULL)
_master_data = data;
} void CMyApplication::run()
{
启动时竞争master,
在成为master之前不能进入工作状态
while (!_stop)
{
int zk_errcode;
std::string zk_errmsg; if (race_master(_master_path, _master_data, &zk_errcode, &zk_errmsg))
{
成为master后,
要让原来的master有足够时间退出master状态
MYLOG_INFO("Race master at %s with %s successfully, sleep for 10 seconds to let the old master quit\n", _master_path.c_str(), _master_data.c_str());
mooon::sys::CUtils::millisleep(10000);
MYLOG_INFO("Start working now\n"); work();
if (!_stop)
{
退出work(),表示需要重新竞争master
MYLOG_INFO("Turn to slave from master at %s with %s successfully, stop working now\n", _master_path.c_str(), _master_data.c_str());
}
}
else
{
如果node_exists_exception()返回true,表示已有master,
即_master_path已存在,返回false为其它错误,应将错误信息记录到日志
if (node_exists_exception(zk_errcode))
{
MYLOG_INFO("A master exists\n");
}
else
{
MYLOG_ERROR("Race master at %s with %s failed: (state:%d)(errcode:%d)%s\n", _master_path.c_str(), _master_data.c_str(), get_state(), zk_errcode, zk_errmsg.c_str());
if (invalid_handle_exception(zk_errcode))
{
MYLOG_INFO("To recreate session\n");
recreate_session();
}
} 休息2秒后再尝试,不要过频重试,一般情况下1~10秒都是可接受的
mooon::sys::CUtils::millisleep(2000);
}
} MYLOG_INFO("Exit now\n");
} void CMyApplication::wait()
{
} void CMyApplication::work()
{
要及时检查is_connected(),以防止master失效后同时存在两个master
while (!_stop && !is_session_expired())
{
mooon::sys::CUtils::millisleep(2000);
MYLOG_INFO("Working with state:\033[1;33m%d\033[m ...\n", get_state());
}
} void CMyApplication::on_zookeeper_session_connected(const char* path)
{
MYLOG_INFO("[\033[1;33mon_zookeeper_session_connected\033[m] path: %s\n", path); const std::string zk_parent_path = "";
const std::string zk_node_name = "test";
const std::string zk_node_data = "123"; try
{
create_node(zk_parent_path, zk_node_name, zk_node_data, ZOO_EPHEMERAL);
MYLOG_INFO("Create %s/%s ok\n", zk_parent_path.c_str(), zk_node_name.c_str());
}
catch (mooon::utils::CException& ex)
{
MYLOG_ERROR("Create %s/%s failed: %s\n", zk_parent_path.c_str(), zk_node_name.c_str(), ex.str().c_str());
}
} void CMyApplication::on_zookeeper_session_connecting(const char* path)
{
MYLOG_INFO("[\033[1;33mon_zookeeper_session_connecting\033[m] path: %s\n", path);
} void CMyApplication::on_zookeeper_session_expired(const char *path)
{
MYLOG_INFO("[\033[1;33mon_zookeeper_session_expired\033[m] path: %s\n", path);
//exit(1); 最安全的做法,在这里直接退出,通过重新启动方式再次竞争master
} void CMyApplication::on_zookeeper_session_event(int state, const char *path)
{
MYLOG_INFO("[\033[1;33mon_zookeeper_session_event\033[m][state:%d] path: %s\n", state, path);
} void CMyApplication::on_zookeeper_event(int type, int state, const char *path)
{
MYLOG_INFO("[\033[1;33mon_zookeeper_event\033[m][type:%d][state:%d] path: %s\n", type, state, path); if (type == 3)
{
const int data_size = mooon::SIZE_4K;
const bool keep_watch = true;
std::string zk_data;
const int n = get_zk_data(path, &zk_data, data_size, keep_watch);
printf("(%d/%zd)%s\n", n, zk_data.size(), zk_data.c_str());
}
}
Zookeeper C++编程实战之主备切换的更多相关文章
- 基于zookeeper的主备切换方法
继承CZookeeperHelper即可快速实现主备切换: https://github.com/eyjian/libmooon/blob/master/include/mooon/net/zooke ...
- Zookeeper C++编程实战之配置更新
CZookeeperHelper:https://github.com/eyjian/libmooon/blob/master/include/mooon/net/zookeeper_helper.h ...
- Spark系列(五)Master主备切换机制
Spark Master主备切换主要有两种机制,之中是基于文件系统,一种是基于Zookeeper.基于文件系统的主备切换机制需要在Active Master挂掉后手动切换到Standby Master ...
- 大话Spark(7)-源码之Master主备切换
Master作为Spark Standalone模式中的核心,如果Master出现异常,则整个集群的运行情况和资源都无法进行管理,整个集群将处于无法工作的状态. Spark在设计的时候考虑到了这种情况 ...
- Nginx+Keepalived主备切换(包含nginx服务停止)
原文地址:http://blog.sina.com.cn/s/blog_79ac6aa80101bmed.html Nginx+Keepalived主备切换(包含nginx服务停止) 环境: VM中4 ...
- 在Windows Azure上配置VM主备切换(1)——Linux篇
对任何一个上线系统来说,高可用设计是不可或缺的一个环节,这样才可以确保应用可以持续.稳定的运行,而不是频繁的掉线.停机.高可用设计的核心思路很简单,就是消除一切单点故障,将单点链路或者节点升级为多点. ...
- (摘)DataGuard物理standby管理 - 主备切换
DataGuard物理standby管理 - 主备切换 Dataguard的切换分为两种,switchover和failover. switchover一般用于数据库或硬件升级,这时只需要较短时间中断 ...
- 测试redis+keepalived实现简单的主备切换【转载】
转自: 测试redis+keepalived实现简单的主备切换 - Try My Best 尽力而为 - ITeye技术网站http://raising.iteye.com/blog/2311757 ...
- Redis安装,主从,主备切换
网络环境: 主:10.187.120.5 从:10.187.69.58 从:10.187.69.59 一.安装 mv redis-2.8.19.tar.gz /export/servers/ cd / ...
随机推荐
- eclipse导入项目以后,内容没有错误,项目上却有个小红叉?
对于上面的错误,应该如何解决?
- js map/reduce
map 由于map()方法定义在JavaScript的Array中,我们调用Array的map()方法,传入我们自己的函数,就得到了一个新的Array作为结果: function pow(x) { r ...
- Python知识
1 注释 单行注释:# 内容 多行注释:A """ ''' 内容 或 内容 "&quo ...
- C# iframe session 丢失
在页面A中使用iframe引用另一站点页面B,但页面B上面的session总是丢失,百度了一下,不用改程序,直接在iis里面操作,解决方法如下 1.打开IIS管理器 inetmgr 2.选择被嵌入if ...
- BOM心得
Brower Objects Model浏览器对象模型 ps: 到现在也没个正式标准.............. window是BOM的顶级对象,但一般可以省略 一.Location对象 相当于浏览器 ...
- SVN查看最新几次提交日志的命令
Windows下,使用svn客户端查看日志很方便,linux下查看的记录条数的小技巧: 1>查看某个目录下的所有日志记录 svn log [PATH] 上面的方式,对于命令行下查看时是很不方便的 ...
- 摹客iDoc201901-2新功能点评
2019才刚刚开始,摹客团队就已经蓄势待发.马不停蹄地给大家带来了又一份惊喜.实话说,这次小摹都忍不住要点个赞!下面就赶紧带大家看看iDoc又更新了哪些新功能: 1.标注和评论融合.协作更高效 iDo ...
- java 内存, 类加载g
1. java 内存区域 方法区 虚拟机栈 本地方法栈 堆 程序计数器 其中 : 方法区 和 堆 是所有线程共享的 , 其他是线程隔离的 1. 程序计数器 : 可以看做是当前线程所执行的字节码的 ...
- 关于部署传统的Dynamic Web项目
现在大部分都是采用maven构建的项目,但是偶尔也会遇到一些较老的项目,采用的是传统的动态Web项目. 我最近碰到这样一个项目,项目用的jar包都放在了WEB-INF/lib目录下.之前的人采用的部署 ...
- Windows 8风格应用-触控输入
参考:演练:创建您的第一个触控应用程序 http://msdn.microsoft.com/zh-cn/library/ee649090(v=vs.110).aspx win8支持多点触摸技术,而我们 ...