1. 基本介绍

在分布式的环境中,可能会有多个对等的程序读取相同的配置文件,程序能够部署在多台机器上,假设配置採用文件的话,则须要为部署该程序的机器也部署一个配置文件,一旦要改动配置的时候就会很麻烦,须要改动多个配置文件,并且easy产生不一致。

集中式配置管理的思路是,将配置数据集中公布到ZooKeeper的节点上。供订阅者动态获取数据。实现配置的集中式管理和动态更新。

能够简单的理解为配置数据与程序分离。

2. 场景分析

(1).集中式配置管理

通常来说,大部分项目里面都有约定的配置文件格式,如ini,xml等。一般都会有相应的解析库类。这样的解析库类的基本工作模式为:

  1. 读取文件(open)
  2. 解析文件(parse)
  3. 对外提供參数(get)

假设我们将文件的内容所有放到ZooKeeper的某个节点上.解析类将配置数据所有下载到本地,在完毕解析的话。则能够用非常小的修改就完毕集中式配置管理的需求。

  1. 读取Zookeeper上相应路径的数据(read)
  2. 解析文件(parse)
  3. 对外提供參数(get)

(2).动态更新

动态更新是希望不重新启动程序就行实时获取更新的配置。在单机环境中,这样的配置数据一般会放在数据库中。改动配置仅仅须要update数据库就行了。

使用ZooKeeper的话,须要节点注冊一个watcher,监视配置数据的是否有变化,一定出现变化,则调用新的解析类来又一次解析配置数据。

个人觉得这个特征使用Zookeeper能够实现,可是并非全部配置都须要这个功能,这样的比較适合对配置敏感,须要实时更新配置的情况。

3. 动手实践

这里我仅仅实现了集中式配置管理的功能,没有实现动态更新,有须要的话你能够尝试自己实现。

因为之前以前做个一个ini文件的库类解析,这里就直接拿过来改了。

依据场景的分析,仅仅须要改动open这个函数就ok了。

看下原来的open函数

/*读取文件名称要改为地址和路径*/
int IniFile::open(const string &filename)
{
release();
fname_ = filename;
IniSection *section = NULL;
/*读取数据的方式须要改动*/
FILE *fp = fopen(filename.c_str(),"r"); if(fp == NULL ){
return -1;
} string line;
string comment; //添加默认段
section = new IniSection();
sections_[""] = section;
/*获取行的方式须要改动*/
while(getline(line,fp) > 0){ ...//省略单行的解析 } fclose(fp); return 0;
}

我们有三个主要须要改动的地方。各自是是入參,fopen和getline。

以下是改动后的open函数

/*改动入參。host为Zookeeper的ip及port地址。filepath为配置数据的路径*/
int IniFile::open2(const string &host,const string &filepath)
{
release();
fname_ = filepath;
IniSection *section = NULL;
char fp[2048]={0};
/*ZooKeeper来读取*/
zkopen(host,filepath,fp,sizeof(fp)); if(fp[0] == 0){
return -1;
} string line;
string comment; //添加默认段
section = new IniSection();
sections_[""] = section; char *p = fp;
/*调整getline的入參*/
while(getline2(line,p) > 0){
...//省略单行的解析 } return 0;
}

zkopen从Zookeeper的节点上读取数据,并保存到fp中。

代码例如以下:

string zkopen(const string &host,const string &filepath,char *fp,int len)
{
int timeout = 30000;
char path_buffer[512];
int bufferlen=sizeof(path_buffer);
char conf_data[2048];
int conf_len=sizeof(conf_data); zoo_set_debug_level(ZOO_LOG_LEVEL_WARN); //设置日志级别,避免出现一些其它信息 zhandle_t* zkhandle = zookeeper_init(host.c_str(),NULL, timeout, 0, (char *)"Monitor Test", 0); if (zkhandle ==NULL)
{
fprintf(stderr, "Error when connecting to zookeeper servers...\n");
exit(EXIT_FAILURE);
} int ret = zoo_get(zkhandle,filepath.c_str(),0,conf_data,&conf_len,NULL);
if(ret != ZOK){
fprintf(stderr,"failed to get the data of path %s!\n",filepath.c_str());
conf_data[0] = 0;
} zookeeper_close(zkhandle); strncpy(fp,conf_data,len);
return conf_data;
}

接下来在对照下调用的变化。

原来的调用方式:

  /** read test **/
IniFile ini;
ini.open(g_filepath); //获取指定段的指定项的值
int ret = 0;
string db_name = ini.getStringValue("COMMON","DB",ret);
string db_passwd = ini.getStringValue("COMMON","PASSWD",ret);

如今的调用方式:

 /** read test **/
IniFile ini;
ini.open2(g_host,g_filepath);/*仅此处有变化*/ //获取指定段的指定项的值
int ret = 0;
string db_name = ini.getStringValue("COMMON","DB",ret);
string db_passwd = ini.getStringValue("COMMON","PASSWD",ret);

由上可见,配置的改造还是非常easy的。并且对程序的修改非常小。

代码详见https://github.com/Winnerhust/ZooKeeper-Exam/tree/master/Config

5.小提示

须要注意一点,配置文件里通常有非常多换行,而ZooKeeper的client命令行工作不支持字符转义。

比方你要将一个配置文件test.ini的内容保存到Zookeeper上,文件内容例如以下。

[COMMON]

DB=mysql

PASSWD=root

你可能会在Zookeeperclient上输入:

[zk: 172.17.0.36:2181(CONNECTED) 39] create /Conf/test.ini [COMMON]\nDB=mysql\nPASSWD=root\n

结果与我们希望的并不一样:

[zk: 172.17.0.36:2181(CONNECTED) 43] get /Conf/test.ini3[COMMON]\nDB=mysql\nPASSWD=root\n

Zookeeper并没有将字符串进行转义,所以不能用ZooKeeperclient直接上传配置文件。因此在代码里我还添加了一个上传配置的功能。仅仅须要将上一个參数-r就能够了。如将test.ini文件的内容上传到ZooKeeper:

cat test.ini | testcase -r -p/Conf/test.ini -s172.17.0.36:2181

ZooKeeper场景实践:(2)集中式配置管理的更多相关文章

  1. springCloud学习1(集中式配置管理)

    springcloud 总集:https://www.tapme.top/blog/detail/2019-02-28-11-33 一.前言   在开发普通的 web 应用中,我们通常是将配置项写在单 ...

  2. ZooKeeper场景实践:(6)集群监控和Master选举

    1. 集群机器监控 这通经常使用于那种对集群中机器状态,机器在线率有较高要求的场景,可以高速对集群中机器变化作出响应.这种场景中,往往有一个监控系统,实时检測集群机器是否存活. 利用ZooKeeper ...

  3. 中小型研发团队架构实践七:集中式日志ELK

    一.集中式日志 日志可分为系统日志.应用日志以及业务日志,系统日志给运维人员使用,应用日志给研发人员使用,业务日志给业务操作人员使用.我们这里主要讲解应用日志,通过应用日志来了解应用的信息和状态,以及 ...

  4. 理解OpenShift(6):集中式日志处理

    理解OpenShift(1):网络之 Router 和 Route 理解OpenShift(2):网络之 DNS(域名服务) 理解OpenShift(3):网络之 SDN 理解OpenShift(4) ...

  5. [hadoop][基本原理]zookeeper场景使用

    代码:https://github.com/xufeng79x/ZkClientTest 1. 简介 zookeeper的特性决定他适用到某些场景非常合适,比如典型的应用场景: 1.集群管理(Grou ...

  6. zookeeper作为soa服务器集群的协调调度服务器

    zookeeper作为soa服务器集群的协调调度服务器,当然自身也支持集群. ZooKeeper搭建系列集 ZooKeeper系列之一:ZooKeeper简介 ZooKeeper系列之二:ZooKee ...

  7. ELK+Filebeat 集中式日志解决方案详解

    链接:https://www.ibm.com/developerworks/cn/opensource/os-cn-elk-filebeat/index.html?ca=drs- ELK Stack ...

  8. 安装logstash+kibana+elasticsearch+redis搭建集中式日志分析平台

    安装logstash+kibana+elasticsearch+redis搭建集中式日志分析平台 2014-01-16 19:40:57|  分类: logstash |  标签:logstash   ...

  9. 通过python构建集中式的病毒扫描机制

    Clam AntiVirus(Clam AV)是一个免费而且开放源码的防毒软件,软件与病毒库的更新由开源社区免费发布,目前ClamdAV主要为Linux.Uinux系统提供病毒扫描查杀pyClamad ...

随机推荐

  1. 六款常用的linux C/C++ IDE

    摘要: 一.AnjutaAnjuta是一个多语言的IDE,它最大的特色是灵活,同时打开多个文件,内嵌代码级的调试器(调用gdb),应用程序向导(Application wizards)可以方便的帮助你 ...

  2. vim 操作指令2

    VIM命令大全 光标控制命令 命令 光标移动 h 向左移一个字符 j 向下移一行 k 向上移一行 l 向右移一个字符 G 移到文件的最后一行 w 移到下一个字的开头 W 移到下一个字的开头,忽略标点符 ...

  3. VS2010(2012)中使用Unit Testing进行单元测试

    原文 VS2010(2012)中使用Unit Testing进行单元测试 使用VS 2012自带的Unit Testing工具进行单元测试是非常方便的.网上关于这方面的例子很多,这篇随笔只起个人学习笔 ...

  4. 获取合并单元格中值的一个方法POI

    private static String getCellValueForMerginRegion(Cell cell) { int rowIdx=cell.getRowIndex(); Sheet ...

  5. SQL 事务及实例演示

    简介 事务,英文名称是transaction.是在对数据库进行管理操作过程中一个逻辑单位,由有限的操作序列构成. 其实这个概念很好懂,简单理解就是:事务就是在使用数据库中的一个操作,由一些操作放到一起 ...

  6. OpenStreetMap初探(一)——了解OpenStreetMap

    1. 開始关注OpenStreetMap始于此博文:<微软对抗谷歌的秘密武器:开源地图OpenStreetMap>  http://news.csdn.net/a/20120328/313 ...

  7. Android菜鸟的成长笔记(8)——Intent与Intent Filter(上)

    原文:[置顶] Android菜鸟的成长笔记(8)——Intent与Intent Filter(上) Intent代表了Android应用的启动“意图”,Android应用将会根据Intent来启动指 ...

  8. tracert路由跟踪命令分析判断

    可能有的会使用路由跟踪命令 ,可是却看不太明确显示出来的结果.结合我的来说明一下. (1)tracert命令介绍 tracert是路由跟踪命令,通过该命令的返回结果,能够获得本地到达目标主机所经过的网 ...

  9. Android 编程之第三方开发 MaoZhuaWeiBo微博开发演示样例-1

    在大学期间我做过非常多类似这种APP.这个是我们小组之前做的,我后期增加非常多新元素.完好了这个应用,由于为了加强 专业技术嘛.也是常常熬夜写些小东西,嘿嘿.只是还算不错.起码技术长进了不少嘛,还是非 ...

  10. SQL Server 数据的添加修改删除和查询

    数据的添加: 首先建立一个数据库,点击新建查询,然后用代码建立一个表,表里写上列名和数据类型,约束可加可不加 然后使用insert语句往表里添加数据 insert [into] 表名 (列名1,列名2 ...