[奇思异想]使用Zookeeper管理数据库连接串

时间:2022-01-19 02:05:05

背景

  有一套特定规格的应用(程序+数据库),当有业务需求时,就需要多部署应用,并且所有的应用都使用一个共同的后台来管理。应用新增后,如何通知后台更新连接串成了一个关键的问题。于是就产生了使用ZooKeeper管理数据库连接串的奇思异想。具体方案如下:

  [奇思异想]使用Zookeeper管理数据库连接串

  1. 运维负责搭建数据库,并执行初始化脚本,然后把对应的数据库配置刷入ZooKeeper;

  2. 运维完成App(1...N)的部署,App(1...N)从ZooKeeper读取对应的数据库配置;

  3. 后台监听ZooKeeper,更新数据库配置到后台应用内存。

环境准备

  1. 安装Zookeeper

  docker pull zookeeper:3.4.13

  [奇思异想]使用Zookeeper管理数据库连接串

  

  docker run --name zookeeper -d -p 2181:2181 zookeeper:3.4.13

  [奇思异想]使用Zookeeper管理数据库连接串

  2. 安装Mysql

  docker pull mysql:5.7

  [奇思异想]使用Zookeeper管理数据库连接串

  docker run --name mysql -e MYSQL_ROOT_PASSWORD=root -p 3306:3306 -d mysql:5.7
  docker run --name mysql2 -e MYSQL_ROOT_PASSWORD=root -p 3307:3306 -d mysql:5.7
  docker run --name mysql3 -e MYSQL_ROOT_PASSWORD=root -p 3308:3306 -d mysql:5.7

  [奇思异想]使用Zookeeper管理数据库连接串

  3. 初始化数据库

CREATE DATABASE test;
USE test;
CREATE TABLE `table` (
`id` int(11) NOT NULL,
`name` varchar(50) NOT NULL,
PRIMARY KEY (`id`)
);

  [奇思异想]使用Zookeeper管理数据库连接串 [奇思异想]使用Zookeeper管理数据库连接串

  分别在各个数据库插入测试数据

  mysql:

USE test;
INSERT INTO `table` (id, name) VALUES (1, 'A1');
INSERT INTO `table` (id, name) VALUES (2, 'B1');
INSERT INTO `table` (id, name) VALUES (3, 'C1');

  mysql2:

USE test;
INSERT INTO `table` (id, name) VALUES (1, 'A2');
INSERT INTO `table` (id, name) VALUES (2, 'B2');
INSERT INTO `table` (id, name) VALUES (3, 'C2');

  mysql3:

USE test;
INSERT INTO `table` (id, name) VALUES (1, 'A3');
INSERT INTO `table` (id, name) VALUES (2, 'B3');
INSERT INTO `table` (id, name) VALUES (3, 'C3');

  4. 基于数据库生成POCO

  Install-Package MySql.Data.EntityFrameworkCore -Version 8.0.13

  Scaffold-DbContext "server=127.0.0.1;port=3306;user=root;password=123456;database=test" MySql.Data.EntityFrameworkCore -OutputDir DataAccess -f

  [奇思异想]使用Zookeeper管理数据库连接串

  [奇思异想]使用Zookeeper管理数据库连接串

  5. 引用ZooKeeper相关组件

  Install-Package ZooKeeperNetEx -Version 3.4.12.1

  

核心代码

  1. ZookeeperOption:从appsettings中读取ZooKeeper相关配置

    public class ZookeeperOption
{
public ZookeeperOption(IConfiguration config)
{
if (config == null)
{
throw new ArgumentNullException(nameof(config));
} var section = config.GetSection("zookeeper");
section.Bind(this);
} public string ConnectionString { get; set; } public int Timeout { get; set; }
}

  2. ZookeeperServiceCollectionExtensions:注册ZooKeeper服务

    public static class ZookeeperServiceCollectionExtensions
{
public static IServiceCollection AddZookeeper(this IServiceCollection services, IConfiguration config)
{
if (services == null)
{
throw new ArgumentNullException(nameof(services));
} if (config == null)
{
throw new ArgumentNullException(nameof(config));
} services.AddOptions(); var option = new ZookeeperOption(config); var zookeeper = new org.apache.zookeeper.ZooKeeper(option.ConnectionString, option.Timeout * , new DefaultWatcher()); services.Add(ServiceDescriptor.Singleton(zookeeper));
return services;
}
} public class DefaultWatcher : Watcher
{
public override Task process(WatchedEvent @event)
{
return Task.CompletedTask;
}
}

  3. ZookeeperHandler:ZooKeeper初始化及目录变化处理类,并把数据库连接信息写入程序内存

    public interface IZookeeperHandler
{
Task InitAsync(); Task RefreshAsync();
} public class ZookeeperHandler: IZookeeperHandler
{
private readonly org.apache.zookeeper.ZooKeeper _zooKeeper;
private readonly IMemoryCache _cache; public ZookeeperHandler(org.apache.zookeeper.ZooKeeper zooKeeper, IMemoryCache cache)
{
_zooKeeper = zooKeeper;
_cache = cache;
} public async Task InitAsync()
{
await RefreshAsync();
} public async Task RefreshAsync()
{
var connDic = new Dictionary<string, string>(); var isExisted = await _zooKeeper.existsAsync("/connections");
if (isExisted == null)
{
await _zooKeeper.createAsync("/connections", null, ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);
} var connResult = await _zooKeeper.getChildrenAsync("/connections", new ConnectionWatcher(this)); foreach (var conn in connResult.Children)
{
var connData = await _zooKeeper.getDataAsync($"/connections/{conn}/value");
var connStr = Encoding.UTF8.GetString(connData.Data);
connDic[conn] = connStr;
} _cache.Set("connections", connDic);
}
}

  4. ConnectionWatcher:监听者,内容变化时调用ZookeeperHandler的RefreshAsync()方法,其中,变化只通知一次,因此需要再次建立监听

    public class ConnectionWatcher : Watcher
{
private readonly IZookeeperHandler _zookeeperService; public ConnectionWatcher(IZookeeperHandler zookeeperService)
{
_zookeeperService = zookeeperService;
} public override async Task process(WatchedEvent @event)
{
var type = @event.get_Type(); if (type != Event.EventType.None)
{
await _zookeeperService.RefreshAsync();
}
}
}

  5. ZookeeperApplicationBuilderExtensions:初始化

    public static class ZookeeperApplicationBuilderExtensions
{
public static IApplicationBuilder UseZookeeper(this IApplicationBuilder app)
{
var service = app.ApplicationServices.GetRequiredService<IZookeeperHandler>();
service.InitAsync().Wait();
return app;
}
}

  6. ContextProvider:根据Id从内存中读取对应的数据库连接串,并提供DbContext实例

    public interface IContextProvider
{
TestContext GetContext(string id);
} public class ContextProvider : IContextProvider
{
private readonly IMemoryCache _cache; public ContextProvider(IMemoryCache cache)
{
_cache = cache;
} public TestContext GetContext(string id)
{
var dic = _cache.Get<Dictionary<string, string>>("connections");
var connectionStr = dic[id]; var optionsBuilder = new DbContextOptionsBuilder<TestContext>();
optionsBuilder.UseMySQL(connectionStr);
return new TestContext(optionsBuilder.Options);
}
}

效果演示

  1. 刚开始没有任何连接信息

  [奇思异想]使用Zookeeper管理数据库连接串

  2. 添加一个连接信息

  [奇思异想]使用Zookeeper管理数据库连接串

  [奇思异想]使用Zookeeper管理数据库连接串

  3. 查询连接对应的数据

  [奇思异想]使用Zookeeper管理数据库连接串

  4. 再添加两个连接信息

  [奇思异想]使用Zookeeper管理数据库连接串

  [奇思异想]使用Zookeeper管理数据库连接串

  [奇思异想]使用Zookeeper管理数据库连接串

  [奇思异想]使用Zookeeper管理数据库连接串

  [奇思异想]使用Zookeeper管理数据库连接串

  5. 查看ZooKeeper的信息

  docker run -it --rm --link zookeeper:zookeeper zookeeper:3.4.13 zkCli.sh -server zookeeper

  ls /connections

  get /connections/1/value

  get /connections/2/value

  get /connections/3/value

  [奇思异想]使用Zookeeper管理数据库连接串

补充说明

  因为把数据库连接信息写到了程序内存中,因此,如果当ZooKeeper出现了故障:

  1. 老的(正在运行)应用正在使用的数据库不会受到影响,但无法监听到数据库信息的变化;

  2. 新的应用无法启动。

  ZooKeeper恢复后:

  1. 老的(正在运行)应用会重连,重新监听到数据库信息的变化

  2. 新的应用可以成功启动。

源码地址

  https://github.com/ErikXu/zookeeper-connection-management

[奇思异想]使用Zookeeper管理数据库连接串的更多相关文章

  1. &lbrack;奇思异想&rsqb;使用RabbitMQ实现定时任务

    背景 工作中经常会有定时任务的需求,常见的做法可以使用Timer.Quartz.Hangfire等组件,这次想尝试下新的思路,使用RabbitMQ死信队列的机制来实现定时任务,同时帮助再次了解Rabb ...

  2. c&num; 扩展方法奇思妙用基础篇八:Distinct 扩展(转载)

    转载地址:http://www.cnblogs.com/ldp615/archive/2011/08/01/distinct-entension.html 刚看了篇文章 <Linq的Distin ...

  3. c&num; 扩展方法奇思妙用

    # 扩展方法出来已久,介绍扩展方法的文章也很多,但都是笼统的.本人最近一直在思考扩展方法的应用,也悟出了一些,准备将这最近一段时间对扩展方法的思考,写成一个系列文章.每个文章只介绍一个应用方面,篇幅不 ...

  4. c&num; 扩展方法奇思妙用基础篇八:Distinct 扩展

    刚看了篇文章 <Linq的Distinct太不给力了>,文中给出了一个解决办法,略显复杂. 试想如果能写成下面的样子,是不是更简单优雅 var p1 = products.Distinct ...

  5. 池化技术(二)HikariCP是如何管理数据库连接的?

    基于依赖程序的版本信息:HikariCP:3.3.1               驱动程序mysql-connector-java:8.0.17 上一篇:Druid是如何管理数据库连接的 零.类图和流 ...

  6. 池化技术(一)Druid是如何管理数据库连接的?

    基于依赖程序的版本信息:druid:1.1.16               驱动程序mysql-connector-java:8.0.17 下一篇:HikariCP是如何管理数据库连接的 零.类图& ...

  7. c&num; 扩展方法奇思妙用集锦

    本文转载:http://www.cnblogs.com/ldp615/archive/2009/08/07/1541404.html 其中本人觉得很经典的:c# 扩展方法奇思妙用基础篇五:Dictio ...

  8. Zookeeper管理多个HBase集群

    zookeeper是hbase集群的"协调器".由于zookeeper的轻量级特性,因此我们可以将多个hbase集群共用一个zookeeper集群,以节约大量的服务器.多个hbas ...

  9. Mysql系列九:使用zookeeper管理远程Mycat配置文件、Mycat监控、Mycat数据迁移(扩容)

    一.使用zookeeper管理远程Mycat配置文件 环境准备: 虚拟机192.168.152.130: zookeeper,具体参考前面文章 搭建dubbo+zookeeper+dubboadmin ...

随机推荐

  1. 解决微信公众号OAuth出现40029&lpar;invalid code,不合法的oauth&lowbar;code&rpar;的错误

    关于OAuth 官方教程:https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1421140842&token=&la ...

  2. java hashCode方法返回值

    hashCode 是和内存地址相关的一个整数. HashCode只是在需要用到哈希算法的数据结构中才有用 用途是为了方便快速地查找对象: HashMap 是根据键对象的 HashCode 来进行快速查 ...

  3. array&lowbar;merge注意细节

    array_merge:合并一个或多个数组,一个数组中的值加在前一个数组的后面,返回的新数组作为结果 如果输入的数组中有相同的字符串键名,则该键名后面的值覆盖前面的,如果数组包含相同的数字键名,后面的 ...

  4. 脱离Xcode,程序在模拟器中无法运行

    今天在调试项目的时候 突然发现,如果项目不通过Xcode启动而是直接通过模拟器进行启动,程序闪一下马上退出,并且不是闪退,而是跑到后台去了,并且后台的程序同样无法启动.找了好多解决办法,最后的解决方案 ...

  5. prmopt 提示框接收字符串,输入后按确定弹出警告框,警告内容为逆序的字符串

    虽然已经找到offer,但因为公司还没安排实习,所以在学校的时间多了很多.好吧,这段时间我用来备考四级啦(好悲催,还没过),然后这一天,闲着无聊,就帮妹妹看了这样子一道题目啦. 题目内容: 编制一个从 ...

  6. svg都快忘了,复习一下

    http://www.360doc.com/content/07/0906/21/39836_724430.shtml

  7. 线段树(区间树)之区间染色和4n推导过程

    前言 线段树(区间树)是什么呢?有了二叉树.二分搜索树,线段树又是干什么的呢?最经典的线段树问题:区间染色:正如它的名字而言,主要解决区间的问题 一.线段树说明 1.什么是线段树? 线段树首先是二叉树 ...

  8. 鼠标交互插件threex&period;domevents介绍

    threex.domevents是一个three.js的扩展库,支持3D场景的交互.和我们操作DOM树的事件相似,名称都是一样的.所以使用起来非常方便.另外他也提供了连接操作.单击网格可实现跳转功能. ...

  9. html5之range

    第一次以这种方式做笔记,希望可以加强自己对新知识的理解,更希望能得到更多朋友的指正. 言归正传: <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.0 ...

  10. 解释JS变量作用域的范例

    JS的变量作用域只有两种:全局作用域与函数作用域. 用var声明的变量不能简单的说是属于函数作用域,应该是说属于其最近的作用域. var a = 10; function test(){ var a; ...