欢迎来到米奇佳佳屋的博客

人生三从境界:昨夜西风凋碧树,独上高楼,望尽天涯路。 衣带渐宽终不悔,为伊消得人憔悴。 众里寻他千百度,蓦然回首,那人却在灯火阑珊处。

datax - 艰难debug路

datax - 艰难debug路

第一次使用datax进行数据同步,bug慢慢啊,所以在此处把自己遇到的问题梳理做个总结,同时,对于datax进行一个简单的介绍

1. datax介绍

​ datax 数据同步离线工具,将不同数据源的同步抽象为从源头数据源读取数据的Reader插件,以及向目标端写入数据的Writer插件,理论上DataX框架可以支持任意数据源类型的数据同步工作。同时DataX插件体系作为一套生态系统, 每接入一套新数据源该新加入的数据源即可实现和现有的数据源互通。

​ 安装部署-安装路径,只需要进行相应的安装解压到相应的路径即可。每次执行任务则是将使用/datax/bin/data.py文件来去执行相应的任务即可。

​ DataX执行同步任务得时候,将数据源读取和写入抽象成为Reader/Writer插件,纳入到整个同步框架中。

  • Reader:Reader为数据采集模块,负责采集数据源的数据,将数据发送给Framework。
  • Writer:Writer为数据写入模块,负责不断向Framework取数据,并将数据写入到目的端。
  • Framework:Framework用于连接reader和writer,作为两者的数据传输通道,并处理缓冲,流控,并发,数据转换等核心技术问题。

2. 从MySQL到HDFS

2.1 准备工作

从Mysql到hdfs中,需要做以下工作:

  • 准备MySQL端的数据,建表

  • 准备hive端数据库和表,在同步之前没有相应的分区,因此还需要在hdfs的hive数据库路径下建立相应的分区 / 也可以将此步骤 和 后面的同步任务写在脚本中,定时运行即可。

  • 准备同步任务的json脚本

  • 同步之后,进行hive端的数据加载。

以下是同步任务的json脚本

{
  "job": {
    "setting": {
      "speed": {
        "channel": 1
      },
      "errorLimit": {
        "record": 0
      }
    },
    "content": [{
      "reader": {
        "name": "mysqlreader",
        "parameter": {
          "username": "数据库用户名",
          "password": "数据库密码",
          "column": [
            "字段1", "字段2"
          ],
          "where": "同步条件",
          "splitPk": "id",
          "connection": [{
            "table": [
              "MySQL端同步表名"
            ],
            "jdbcUrl": [
              "jdbc:mysql://数据库IP:数据库端口/对应的数据库"
            ]
          }]
        }
      },
      "writer": {
        "name": "hdfswriter",
        "parameter": {
          "defaultFS": "hdfs://HDFS集群-IP:端口",
          "fileType": "text",
          "path": "hive端表对应的存储路径/dt=分区参数(分区按照实际情况可要可不要)",
          "fileName": "hive端表名$do_date.dat",
          "column": [{
            "name": "字段1",
            "type": "INT"
          },
            {
              "name": "字段2",
              "type": "STRING"
            }
          ],
          "writeMode": "append",
          "fieldDelimiter": "分割符"
        }
      }
    }]
  }
}

2.2 参数说明

setting 参数说明

​ 用来控制同步任务,包括并发通道channel 、字节流byte 、记录流record 、同步速度speed

reader参数说明

        jdbcUrl    描述:数据库地址。必填
        username   描述:数据源用户名
        password   描述: 数据源指定用户名的密码
        table      描述: 所选取的需要同步的表
        column     描述:  所配置的表中需要同步的列名集合
        splitPk    描述:MysqlReader进行数据抽取时,如果指定splitPk,表示用户希望使用splitPk代表的字段进行数据分片,DataX因此会启动并发任务进行数据同步,这样可以大大提供数据同步的效能。推荐splitPk用户使用表主键,因为表主键通常情况下比较均匀,因此切分出来的分片也不容易出现数据热点。splitPk仅支持整形数据切分,不支持浮点、字符串、日期等其他类型。如果用户指定其他非支持类型MysqlReader将报错
        where      描述:筛选条件,MysqlReader根据指定的column、table、where条件拼接SQL,并根据这个SQL进行数据抽取。在实际业务场景中,往往会选择当天的数据进行同步,可以将where条件指定为gmt_create > $bizdate 。注意:不可以将where条件指定为limit 10,limit不是SQL的合法where子句.where条件可以有效地进行业务增量同步。如果不填写where语句,包括不提供where的key或者value,DataX均视作同步全量数据。
        querySql   描述:在有些业务场景下,where这一配置项不足以描述所筛选的条件,用户 可以通过该配置型来自定义筛选SQL。当用户配置了这一项之后,DataX系统就会忽略table,column这些配置型,直接使用这个配置项的内容对数据进行筛选,例如需要进行多表join后同步数据,使用select a,b from table_a join table_b on table_a.id = table_b.id 。当用户配置querySql时,MysqlReader直接忽略table、column、where条件的配置,querySql优先级大于table、column、where选项。

writer参数说明

    defaultFS   描述:Hadoop hdfs文件系统namenode节点地址。格式:hdfs://ip:端口;例如:hdfs://127.0.0.1:9000
    fileType    描述:文件的类型,目前只支持用户配置为"text"或"orc"。 
    path        描述:存储到Hadoop hdfs文件系统的路径信息
    fileName    描述:HdfsWriter写入时的文件名,实际执行时会在该文件名后添加随机的后缀作为每个线程写入实际文件名。 
    column      描述:写入数据的字段,不支持对部分列写入。为与hive中表关联,需要指定表中所有字段名和字段类型,其中:name指定字段名,type指定字段类型。 
    writeMode   描述:hdfswriter写入前数据清理处理模式: 
                § append,写入前不做任何处理,DataX hdfswriter用filename写入,并保证文件名不冲突。
                § nonConflict,如果目录下有fileName前缀的文件,直接报错。
    fieldDelimiter  描述:hdfswriter写入时的字段分隔符,需要用户保证与创建的Hive表的字段分隔符一致,否则无法在Hive表中查到数据 

2.3 执行任务

在做好所有的准备工作的时候,开始执行任务:

#!/bin/bash
source /etc/profile
source activate python27

do_date="`date +%Y-%m-%d`"

# 创建目录
hdfs dfs -mkdir -p /user/hive/warehouse/xxx/xxx/dt=$do_date
# 数据迁移
python2 $DATAX_HOME/datax.py -p "-Ddo_date=$do_date" JSON脚本
# 加载数据
hive -e "alter table xxx库.xxx表  add partition(dt='$do_date')"

3.出现的问题

问题1:数据库连接问题

​ 第一: 地址错误 使用IP

​ 第二: 帐户和密码错误

​ 第三: 表名错误 / 字段问题

​ 对于这种问题,则是进行核查就好

问题2:HDFS 连接问题

​ 第一: HDFS地址问题 ,我犯了最最低级的问题,最开始使用了主机名,而没有用IP

​ 第二:就是连接不上,可以反复重启一下集群解决

问题3:内存不够的问题,报了以下错误

# There is insufficient memory for the Java Runtime Environment to continue.
# Native memory allocation (malloc) failed to allocate 160088 bytes for AllocateHeap
# An error report file with more information is saved as:
# /users/xxx/hs_err_pidxxxx.log

​ 出现这种问题,说明申请内存失败,遇到这种问题,可能由以下两种原因

  1. 系统物理内存或虚拟内存不足

  2. 程序在压缩指针模式下运行, Java堆会阻塞本地堆的增长

    首先使用free-m 查询,查看内存使用情况,如下图所示已经是设置过后的图,剩余内存也不是很多,针对内存过小的问题,通过修改配置文件即可。

image-20210108110229826

由于/proc/meminfo下的vm.overcommit_memory被设置成不允许overcommit造成的overcommit是指对虚拟内存的过量分配; 用户进程申请的是虚拟地址,虚拟内存需要物理内存做支撑, 如果分配太多虚拟内存, 会对性能参数影响.

vm.overcommit_memory的用处: 控制过量分配的策略. 这个参数一共有3个可选值:

0: Heuristic overcommit handling. 就是由操作系统自己决定过量分配策略
1: Always overcommit. 一直允许过量分配
2: Don't overcommit. 不允许过量分配

通过sysctl vm.overcommit_memory 来查看overcommit_memory的参数设置

则通过以下几种方式进行设置:

  1. /etc/sysctl.conf 文件中添加一行vm.overcommit_memory=1 即可,之后再使用sysctl -p 使配置文件永久生效当然这是我们在开发环境下的解决方式, 在生产环境还是要尽量去优化调整JVM的参数来保证每个程序都有足够的内存来保证运行.
  2. echo 1 > /proc/sys/vm/overcommit_memory 此方式临时生效,系统重启后消失
  3. sudo sysctl vm.overcommit_memory=0, 即vm.overcommit_memory = 0, 允许系统自己决定过量分配策略

问题4:导入hive之后数据全为NULL

​ 出现这种问题,主要是分隔符的问题,在hive端建表的时候,设置分隔符最好和JSON脚本中的一致,在网上查了好多情况,有的说是要使用'\t' 分隔符,我测试的结果是我和JSON脚本中设置成一致成功了。

CREATE EXTERNAL
TABLE `ods_trade_shops` (
       `shopId` int,
       `userId` int
)
PARTITIONED by
(
    dt string
)
row format delimited fields terminated by ',';

问题5:在建表的时候需要注意以下,数据类型的问题。

MySQL数据类型 Datax数据类型 HIVE数据类型
int, tinyint, smallint, mediumint, int, bigint LONG int/tinyint/bigint/smallint
float, double, decimal DOUBLE float double
varchar, char, tinytext, text, mediumtext, longtext, year STRING string
date, datetime, timestamp, time DATE date/ timestamp/string
bit , bool BOOLEAN boolean

DATAX在转换MySQL datatime字段类型为hive的timestamp时会出现问题:datax和hive不支持datetime类型,因此可能回出现同步之后时间字段多/少8小时。
解决办法有两个:
1、转换为string类型;
2、继续用timestamp类型,但是需要行存储(即text存储)。

遇见时间类型转换问题时要小心,保守最好是string,简单的比较大小不会影响后续计算。

posted @ 2021-01-10 14:13  菜鸟码代码  阅读(1171)  评论(0编辑  收藏  举报