【原理】你知道 Change Data Capture 是什么吗?

Change Data Capture(缩写为 CDC)——大概可以机翻为 “变动数据捕获”——你可以将它视为和数据库有关的架构设计模式的一种。它的核心思想是,监测并捕获数据库的变动(包括数据或数据表的插入,更新,删除等),将这些变更按发生的顺序完整记录下来,写入到消息中间件中以供其他服务进行订阅及消费。

适用场景

我们可以把 CDC 认为是数据库事件驱动的一种数据 / 信息分发系统,CDC 主要适用于以下的场景:

异构数据库之间的数据同步或备份 / 建立数据分析计算平台

在 MySQL,PostgreSQL,MongoDB 等等数据库之间互相同步数据,或者把这些数据库的数据同步到 Elasticsearch 里以供全文搜索,当然也可以基于 CDC 对数据库进行备份。而数据分析系统可以通过订阅感兴趣的数据表的变更,来获取所需要的分析数据进行处理,不需要把分析流程嵌入到已有系统中,以实现解耦。

微服务之间共享数据状态

在微服务大行其道的今日,微服务之间信息共享一直比较复杂,CDC 也是一种可能的解决方案,微服务可以通过 CDC 来获取其他微服务数据库的变更,从而获取数据的状态更新,执行自己相应的逻辑。

更新缓存 / CQRS 的 Query 视图更新

通常缓存更新都比较难搞,可以通过 CDC 来获取数据库的数据更新事件,从而控制对缓存的刷新或失效。

而 CQRS 是什么又是一个很大的话题,简单来讲,你可以把 CQRS 理解为一种高配版的读写分离的设计模式。举个例子,我们前面讲了可以利用 CDC 将 MySQL 的数据同步到 Elasticsearch 中以供搜索,在这样的架构里,所有的查询都用 ES 来查,但在想修改数据时,并不直接修改 ES 里的数据,而是修改上游的 MySQL 数据,使之产生数据更新事件,事件被消费者消费来更新 ES 中的数据,这就基本上是一种 CQRS 模式。而在其他 CQRS 的系统中,也可以利用类似的方式来更新查询视图。

整体架构

数据源

数据库(即数据源)需要能完整地记录或输出数据库的事件(数据或数据表的变动)。例如 MySQL 的 binlog,MySQL 通过 binlog(二进制日志)记录数据库的变动事件,而 MySQL 本身可以借助 binlog 来实现主从复制。其实 CDC 本身也只是把这种机制扩展了一下,使之能够作为更广泛的用途。

大部分数据库都有类似的机制:

发布服务(生产者)

这东西现在没有一个明确的名称,这里就暂且称为发布服务。它的功能是解析数据源输出的流式数据,序列化成统一的封装格式,并输出到数据总线中。

Maxwell

zendesk/maxwell,专为 MySQL 数据源使用,它能够读取 binlog,将数据库变更序列化为 JSON,输出到 Kafka, Kinesis, RabbitMQ, Google Cloud Pub/Sub, Redis 等消息队列中。它还提供了 bootstrap 命令,支持将某张表全量数据输出作为起始数据。

Debezium

debezium/debezium,目前支持监听 MySQL,MongoDB,PostgreSQL,Oracle,SQL Server 等数据库的变化,但消息中间件只能使用 Kafka。它的数据交换默认采用 Avro 格式,这种格式对数据 schema 的变更比较友好。目前还处在开发阶段,所以文档不是特别完善,但似乎已经有勇敢的公司用在生产环境中了

Canal

alibaba/canal,它能解析 binlog 并输出,可以利用它提供的 client 来实现自己的订阅消费逻辑,~~ 但这个项目不涉及到输出到通用的消息中间件,因此适用范围较窄。~~ 最近看了一下发现有了更新,支持输出到 kafka,甚至还带一些监控功能,看起来很强。

自行实现

你需要的是一个能够解析数据库输出流的库,比如 noplay/python-mysql-replication 这种,基于这样的库来实现发布服务。

我曾经写过 py-mysql-elasticsearch-sync 这样一个工具,就是为了解决把 MySQL 的数据同步到 Elasticsearch 这样的任务,可以把这东西视为弱鸡版的 CDC(因为中间没有通过数据总线发布,而是直接写入 ES 了)。其中就使用了 python-mysql-replication 这个库来获取数据库变更,它的原理就是将自己伪装成一个 MySQL Slave 数据库,通过解析 binlog 获取数据变更的事件。(和 canal 原理上差不多)

数据总线(消息中间件)

数据总线这部分是一个消息中间件,一般会选 Kafka,RabbitMQ,Redis 之类的,但大部分做 CDC 都会选用 Kafka。

Kafka 官方并不把自己称为消息队列,而是叫 “distributed streaming platform”,的确,目前 kafka 的能力以及演化方向更像是一个数据处理的平台了,而不仅仅是消息中间件,你可以利用 Kafka Streams 来进行一些数据处理。

在整个 CDC 里,Kafka 会作为核心的数据交换组件,或者你可以把它称为数据总线,kafka 集群的健壮性和吞吐量能够支撑海量数据的 pub/sub,并且能够将写入的数据持久化一段时间,发布服务将数据库任何数据变动写入 Kafka,由不同的消费者在上面同时进行订阅和消费,如果有需要,消费者随时可以把自己的 offset 往前调,对以往的消息重新消费。

消费者

消费者部分根据场景而不同,通常只要使用相应消息队列的客户端库实现消费者,根据生产者生成的消息格式进行解析处理即可,例如你用 Kafka 作为队列,那么在 Python 中可以用 confluentinc/confluent-kafka-python 之类的库来进行实现。
而需求比较简单的话可以使用直接现成的开源工具,比如 Kafka 的一些 Sink Connectors

如果有特殊的需求,可以自行实现类似的生产者,消费者(或者借助已有的工具进行改造),将数据变动通过 Kafka 之类的消息总线进行发布,来自定义自己的 CDC 系统。

参考

 
发布了22 篇原创文章 · 获赞 11 · 访问量 1万+

猜你喜欢

转载自blog.csdn.net/t1g2q3/article/details/103754140