sqoop是apache旗下一款“Hadoop和关系数据库服务器之间传送数据”的工具,除此之外还有DataX。

导入数据: MySQL,Oracle导入数据到Hadoop的HDFS、HIVE、HBASE等数据存储系统;

导出数据: 从Hadoop的文件系统中导出数据到关系数据库。

底层工作机制:

将导入或导出命令翻译成 MapReduce 程序来实现,

在翻译出的 MapReduce 中主要是对 InputFormat 和 OutputForm。

Cloudera Manager 支持的版本为 Sqoop 2。

一、添加sqoop2到CDH

sqoop2将Oracle数据导入HDFS_全量导入

因为我使用的的CDH搭建hadoop集群,所以可以通过cloudera manager添加上sqoop2.

二、把Oracle驱动拷贝到Sqoop安装路径中的lib文件夹下

我的ORACLE 是10G ,找到ojdbc6.jar,放到sqoop 的lib 下面,目录是

/opt/cloudera/parcels/CDH-5.10.0-1.cdh5.10.0.p0.41/lib/sqoop/lib 

/opt/cm-5.10.0/share/cmf/lib/ojdbc6.jar

/var/lib/sqoop2/ojdbc6.jar

三、oracle全量数据导入到hdfs

1、root 用户执行:

sudo -u hdfs hadoop fs -mkdir   /data/zhao/

在hdfs的data下,创建一个zhao文件夹

2、切换到hdfs 用户:

su hdfs

执行 export ACCUMULO_HOME=/data

3、执行import操作

sqoop import --connect jdbc:oracle:thin:@192.168.183.251:1521:YJDSJ --username yjdsj --password yjdsj#123 --table T_LOADS_TODAY --target-dir /data/zhao/heng  --m 1 --columns obj_id,today_load,dept_id

说明:

导入的目录若已经存在则要先删除已经存在的目录,否则会报路径以存在的错误提示:

FileAlreadyExistsException: Output directoryhdfs://master:8020/user/operate already exists

所以 /data/zhao/heng 这个目录是不存在的

**hdfs 用户执行该命令删除目录 : **

hadoop fs -rm -r /data/zhao/heng

4、查看

sqoop2将Oracle数据导入HDFS_定时增量导入_02

当出现这个页面,表示导入成功

访问HDFS:http://192.168.43.36:50070/

sqoop2将Oracle数据导入HDFS_全量导入_03

四、定时增量导入

数据全量导入到hdfs后,每隔一段时间oracle又会出现新的数据,此时不可能再将所有数据重新导一遍,我们就需要增量数据导入这一模式了。

​sqoop支持两种增量导入模式:​

1、append:只有新增数据

比如订单表,每个订单有一个唯一标识自增列ID,在关系型数据库中以主键形式存在。之前已经将id在0~100之间的编号的订单导入到这里为HDFS,现在一段时间后我们需要将近期产生的新的订单数据导入HDFS,以供后续数仓进行分析。此时我们只需要指定–incremental 参数为append,–last-value参数为100即可。表示只从id大于100后开始导入。

参数

说明

–incremental append

基于递增列的增量导入(将递增列值大于阈值的所有数据增量导入Hadoop)

–check-column

递增列(int)

–last-value

阈值(int)

例如: sqoop import \
--connect jdbc:mysql://h3:3306/ry \
--username root \
--password haitao.211123 \
--table doit_jw_stu_zsgl \
--hive-import \
--hive-table yiee_dw.doit_jw_stu_zsgl \
--split-by id \
--incremental append \
--check-column id \
--last-value 40 \
-m 2

2、lastmodified : 包含修改和新增数据

此方式要求原有表中有time字段,它能指定一个时间戳,让Sqoop把该时间戳之后的数据导入至HDFS。因为后续订单可能状态会变化,变化后time字段时间戳也会变化,此时Sqoop依然会将相同状态更改后的订单导入HDFS,当然我们可以指定merge-key参数为pid,表示将后续新的记录与原有记录合并。

参数

说明

–incremental append

基于时间列的增量导入(将时间列大于等于阈值的所有数据增量导入Hadoop)

–check-column

时间列(int)

–last-value

阈值(int)

–merge-key

合并列(主键,合并键值相同的记录)

例如: sqoop import \
--connect jdbc:mysql://h3:3306/ry \
--username root \
--password haitao.211123 \
--table doit_jw_stu_zsgl \
--target-dir '/sqoopdata/doit_jw_stu_zsgl' \
--incremental lastmodified \
--check-column update_time \
--last-value '2020-03-18 23:59:59' \
--fields-terminated-by ',' \
-m 1 \

因为我的表中,pid为随机产生的32位数字,所以选择使用lastmodified 。

执行导入:

sqoop import --connect jdbc:oracle:thin:@10.68.24.80:1521:YJDSJ --username yjdsj --password yjdsj#123 --table T_POWERGRID_DAMAGE_FORECAST --target-dir /data/06/02 --m 1  --incremental lastmodified  --check-column CREATE_DATE --merge-key pid  --last-value '2018-6-12 11:00:00'

若定时增量导入,则需要创建一个job

sqoop job --create dashuju -- import --connect jdbc:oracle:thin:@10.68.24.80:1521:YJDSJ --username yjdsj --password yjdsj#123 --table T_POWERGRID_DAMAGE_FORECAST --target-dir /data/06/02 --m 1  --incremental lastmodified  --check-column CREATE_DATE --merge-key pid  --last-value '2018-6-12 11:00:00'

使用crontab定时器

crontab -e

增加定时任务(每35分钟执行一次)

*/35 * * * *  sqoop job --exec dashuju

重启crontab定时器

service crond start

期间启动了多个定时任务,曾报maxContainerCapability 设置不足

异常:

REDUCE capability required is more than the supported max container capability in the cluster. Killing the Job. reduceResourceRequest: <memory:12160, vCores:1> maxContainerCapability:<memory:8192, vCores:24>

需要调整两个参数:

yarn.nodemanager.resource.memory-mb
yarn.scheduler.maximum-allocation-mb

sqoop2将Oracle数据导入HDFS_全量导入_04sqoop2将Oracle数据导入HDFS_定时增量导入_05

每次执行sqoop job时,会提示输入密码,如何避免呢?

====修改配置文件====

vi /etc/sqoop/conf/sqoop-site.xml

<property>
<name>sqoop.metastore.client.record.password</name>
<value>true</value>
<description>If true, allow saved passwords in the metastore.
</description>
</property>

五、hdfs导出到oracle

export工具,是将HDFS平台的数据,导出到外部的结构化存储系统中,可能会为一些应用系统提供数据支持.

全量导出

sqoop export --connect jdbc:oracle:thin:@10.68.24.80:1521:YJDSJ --username yjdsj --password yjdsj#123 --table T_POWERGRID_DAMAGE_FORECAST --export-dir /data/06/03/part-m-00000  --input-fields-terminated-by ',' --input-null-string 'null' --input-null-non-string 'null' -m 1 --batch

此方式要求导出的oracle数据库中的表为空表或者新表,不然新的数据产生,再次导出的时候,会违反主键唯一的约束,可以使用以下的命令:

更新模式

sqoop export --connect jdbc:oracle:thin:@10.68.24.80:1521:YJDSJ --username yjdsj --password yjdsj#123 --table T_POWERGRID_DAMAGE_FORECAST --export-dir /data/06/03/part-m-00000  --input-fields-terminated-by ',' --input-null-string 'null' --input-null-non-string 'null' -m 1 --update-mode allowinsert --update-key PID --batch

–update-mode < mode > 指定更新策略,包括:updateonly(默认)、allowinsert;

如果选择updateonly,只会对mysql中已存在的id数据进行更新,不存在的id数据不会插入了;

如果选择allowinsert,既会更新已存在id数据,也会插入新的id数据

–update-key < col-name > 更新参考的列名称,多个列名使用逗号分;