数仓 | 几种常见的数据同步方式

大数据
数据仓库的特性之一是集成,即首先把未经过加工处理的、不同来源的、不同形式的数据同步到ODS层,一般情况下,这些ODS层数据包括日志数据和业务DB数据。

[[428501]]

本文转载自微信公众号「大数据技术与数仓 」,作者西贝。转载本文请联系大数据技术与数仓公众号。

写在前面

数据仓库的特性之一是集成,即首先把未经过加工处理的、不同来源的、不同形式的数据同步到ODS层,一般情况下,这些ODS层数据包括日志数据和业务DB数据。对于业务DB数据而言(比如存储在MySQL中),将数据采集并导入到数仓中(通常是Hive或者MaxCompute)是非常重要的一个环节。

那么,该如何将业务DB数据高效准确地同步到数仓中呢?一般企业会使用两种方案:直连同步与实时增量同步(数据库日志解析)。其中直连同步的基本思路是直连数据库进行SELECT,然后将查询的数据存储到本地文件作为中间存储,最后把文件Load到数仓中。这种方式非常的简单方便,但是随着业务的发展,会遇到一些瓶颈,具体见下文分析。

为了解决这些问题,一般会使用实时增量的方式进行数据同步,其基本原理是CDC (Change Data Capture) + Merge,即实时Binlog采集 + 离线处理Binlog还原业务数据这样一套解决方案。

本文主要包括以下内容,希望对你有所帮助

  • 常见数据同步方式
  • 流式数据集成

数据同步的方式

直连同步

直连同步是指通过定义好的规范接口API和基于动态链接库的方式直接连接业务库,比如ODBC/JDBC等规定了统一的标准接口,不同的数据库基于这套标准提供规范的驱动,从而支持完全相同的函数调用和SQL实现。比如经常使用的Sqoop就是采取这种方式进行批量数据同步的。

直连同步的方式配置十分简单,很容易上手操作,比较适合操作型业务系统的数据同步,但是会存在以下问题:

  • 数据同步时间:随着业务规模的增长,数据同步花费的时间会越来越长,无法满足下游数仓生产的时间要求。
  • 性能瓶颈:直连数据库查询数据,对数据库影响非常大,容易造成慢查询,如果业务库没有采取主备策略,则会影响业务线上的正常服务,如果采取了主备策略,虽然可以避免对业务系统的性能影响,但当数据量较大时,性能依然会很差。

日志解析

所谓日志解析,即解析数据库的变更日志,比如MySQL的Binlog日志,Oracle的归档日志文件。通过读取这些日志信息,收集变化的数据并将其解析到目标存储中即可完成数据的实时同步。这种读操作是在操作系统层面完成的,不需要通过数据库,因此不会给源数据库带来性能上的瓶颈。

数据库日志解析的同步方式可以实现实时与准实时的同步,延迟可以控制在毫秒级别的,其最大的优势就是性能好、效率高,不会对源数据库造成影响,目前,从业务系统到数据仓库中的实时增量同步,广泛采取这种方式。当然,这种方式也会存在一些问题,比如批量补数时造成大量数据更新,日志解析会处理较慢,造成数据延迟。除此之外,这种方式比较复杂,投入也较大,因为需要一个实时的抽取系统去抽取并解析日志,下文会对此进行详细解释。

如上图所示架构,在直连同步基础之上增加了流式同步的链路,经过流式计算引擎把相应的 Binlog 采集到 Kafka,同时会经过一个 Kafka 2Hive 的程序把它导入到原始数据,再经过一层 Merge,产出下游需要的 ODS 数据。

上述的数据集成方式优势是非常明显的,把数据传输的时间放到了 T+0 这一天去做,在第二天的时候只需要去做一次 merge 就可以了。非常节省时间和计算资源。

流式数据集成实现

实现思路

首先,采用Flink负责把Kafka上的Binlog数据拉取到HDFS上,生成增量表。

然后,对每张ODS表,首先需要一次性制作快照(Snapshot),把MySQL里的存量数据读取到Hive上,这一过程底层采用直连MySQL去Select数据的方式,可以使用Sqoop进行一次性全量导入,生成一张全量表。

最后,对每张ODS表,每天基于存量数据和当天增量产生的Binlog做Merge,从而还原出业务数据。

Binlog是流式产生的,通过对Binlog的实时采集,把部分数据处理需求由每天一次的批处理分摊到实时流上。无论从性能上还是对MySQL的访问压力上,都会有明显地改善。Binlog本身记录了数据变更的类型(Insert/Update/Delete),通过一些语义方面的处理,完全能够做到精准的数据还原。

关于Binlog解析部分,可以使用canal工具,采集到Kafka之后,可以使用Flink解析kafka数据并写入到HDFS上,解析kafka的数据可以使用Flink的DataStreamAPI,也可以使用FlinkSQL的canal-json数据源格式进行解析,使用FlinkSQL相对来说是比较简单的。下面是canal-json格式的kafka数据源。

  1. CREATE TABLE region ( 
  2.   id BIGINT
  3.   region_name STRING 
  4. WITH ( 
  5.  'connector' = 'kafka'
  6.  'topic' = 'mydw.base_region'
  7.  'properties.bootstrap.servers' = 'kms-3:9092'
  8.  'properties.group.id' = 'testGroup'
  9.  'format' = 'canal-json' , 
  10.  'scan.startup.mode' = 'earliest-offset'  
  11. ); 

数据解析完成之后,下面的就是合并还原完整数据的过程,关于合并还原数据,一种比较常见的方式就是全外连接(FULL OUTER JOIN)。具体如下:

生成增量表与全量表的Merge任务,当天的增量数据与昨天的全量数据进行全外连接,该Merge任务的基本逻辑是:

  1. INSERT OVERWRITE TABLE user_order PARTITION(ds='20211012'
  2. SELECT  CASE    WHEN n.id IS NULL THEN o.id  
  3.                 ELSE n.id  
  4.         END 
  5.         ,CASE    WHEN n.id IS NULL THEN o.create_time  
  6.                  ELSE n.create_time  
  7.          END 
  8.         ,CASE    WHEN n.id IS NULL THEN o.modified_time 
  9.                  ELSE n.modified_time  
  10.          END 
  11.         ,CASE    WHEN n.id IS NULL THEN o.user_id  
  12.                  ELSE n.user_id  
  13.          END 
  14.          
  15.         ,CASE    WHEN n.id IS NULL THEN o.sku_code  
  16.                  ELSE n.sku_code  
  17.          END 
  18.         ,CASE    WHEN n.id IS NULL THEN o.pay_fee 
  19.                  ELSE n.pay_fee  
  20.          END 
  21. FROM    ( 
  22.             SELECT  * 
  23.             FROM    user_order_delta 
  24.             WHERE   ds = '20211012' 
  25.             AND     id IS NOT NULL 
  26.             AND     user_id IS NOT NULL 
  27.         ) n 
  28. FULL OUTER JOIN (-- 全外连接进行数据merge 
  29.                     SELECT  * 
  30.                     FROM    user_order 
  31.                     WHERE   ds = '20211011' 
  32.                     AND     id IS NOT NULL 
  33.                     AND     user_id IS NOT NULL 
  34.                  
  35.                 ) o 
  36. ON      o.id = n.id 
  37. AND     o.user_id = n.user_id 

经过上述步骤,即可将数据还原完整。

总结

本文首先介绍了数据仓库构建ODS层常见的数据同步方式,并对每种方式进行了解释,给出了相对应的示意图。接着给出了CDC+Merge的数据同步方案。值得注意的是,Flink1.11引入了CDC的connector,比如MySQL CDC和Postgres CDC,同时对Kafka的Connector支持canal-json和debezium-json以及changelog-json的format,通过这种方式可以很方便地捕获变化的数据,大大简化了数据处理的流程和数据同步的复杂度。

 

责任编辑:武晓燕 来源: 大数据技术与数仓
相关推荐

2010-03-31 16:28:11

Oracle数据库

2022-07-26 15:38:58

数据仓数据治理数据团队

2019-02-26 14:39:20

Windows后门漏洞

2019-04-09 21:10:23

iOS加密框架

2021-07-27 15:40:39

Python数据清洗函数

2018-11-07 09:01:13

Tomcat部署方式

2018-10-10 10:23:53

数据库RedisNoSQL

2022-02-18 09:02:04

数据仓库治理

2023-11-23 16:53:56

数据仓库大数据

2022-08-22 17:46:56

虚拟数仓Impala

2010-07-30 09:16:24

Flex数据绑定

2023-10-30 11:53:37

继承JS父类

2020-04-03 10:30:50

MySQL数据库技术

2021-12-02 08:41:30

数仓建模设计

2021-01-31 23:54:23

数仓模型

2021-06-11 07:26:16

数据仓库机器学习

2021-05-07 16:19:36

异步编程Java线程

2023-03-08 07:50:57

企业数据治理

2021-01-19 11:56:19

Python开发语言

2010-09-25 14:48:55

SQL连接
点赞
收藏

51CTO技术栈公众号