• hive建表指定列分隔符为多字符分隔符实战(默认只支持单字符)


    1、背景:

            后端日志采集完成,清洗入hive表的过程中,发现字段之间的单一字符的分割符号已经不能满足列分割需求,因为字段值本身可能包含分隔符。所以列分隔符使用多个字符列分隔符迫在眉睫。

    hive在建表时,通常使用
    ROW FORMAT DELIMITED
    FIELDS TERMINATED BY "|#"  来限定数据中各个字段的分隔符,这种方式只支持单个分隔符,即:实际只会按照"|"进行分割,

    默认情况下,Hive对于分隔符只支持单字符,不过Hive自带一个工具jar包,这个包支持正则和多字符方式定义分隔符。

    hive从0.14版本以后支持MultiDelimitSerDe,可以比较优雅多解决多分隔符问题。

    ROW FORMAT SERDE 'org.apache.hadoop.hive.contrib.serde2.MultiDelimitSerDe' WITH SERDEPROPERTIES ("field.delim"="|#$")

    2、官方对列支持多个字符的分隔符调研 

    参考MultiDelimitSerDe - Apache Hive - Apache Software Foundation

    Introduction:

    Introduced in HIVE-5871, MultiDelimitSerDe allows user to specify multiple-character string as the field delimiter when creating a table.

    Version:

    Hive 0.14.0 and later.

    Hive SQL Syntax:

    You can use MultiDelimitSerDe in a create table statement like this:

    CREATE TABLE test (
     id string,
     hivearray array,
     hivemap map) 
    ROW FORMAT SERDE 'org.apache.hadoop.hive.serde2.MultiDelimitSerDe'                  
    WITH SERDEPROPERTIES ("field.delim"="[,]","collection.delim"=":","mapkey.delim"="@");

    where field.delim is the field delimiter, collection.delim and mapkey.delim is the delimiter for collection items and key value pairs, respectively. 

    HIVE-20619 moved MultiDelimitSerDe to hive.serde2 in release 4.0.0, so user won't have to install hive-contrib JAR into the HiveServer2 auxiliary directory.

    Limitations:

    • Among the delimiters, field.delim is mandatory and can be of multiple characters, while collection.delim and mapkey.delim is optional and only support single character.
    • Nested complex type is not supported, e.g. an Array.
    • To use MultiDelimitSerDe prior to Hive release 4.0.0, you have to add the hive-contrib jar to the class path, e.g. with the add jar command.

    3、小试牛刀,报错

    建表时直接使用 

    ROW FORMAT SERDE 'org.apache.hadoop.hive.contrib.serde2.MultiDelimitSerDe' WITH SERDEPROPERTIES ("field.delim"="|#$")

    执行查询报错:Class org.apache.hadoop.hive.contrib.serde2.MultiDelimitSerDe not found
    根据网上的方案:进入hive执行(jar包路径根据自己环境路径对应修改)

    1. add jar /usr/hdp/3.1.5.0-152/hive/lib/hive-contrib.jar
    2. CREATE EXTERNAL TABLE `table_tset`(
    3. id string,
    4. name string,
    5. year string)
    6. PARTITIONED BY (
    7. `year` string)
    8. ROW FORMAT SERDE 'org.apache.hadoop.hive.contrib.serde2.MultiDelimitSerDe' WITH SERDEPROPERTIES ("field.delim"="|#$")
    9. LOCATION
    10. 'hdfs://CMBHHA/apps/hive/datahouse/test/table_tset'

    4、几经搜索,问题终于解决,附上完整示例

    4.1、找到和hive版本对应的hive-contrib包,下载jar上传到hdfs上

    org/apache/hive/hive-contrib所有版本Jar文件及下载 -时代Java

    hive-contrib包下载链接

    4.2、以三个等号作为列分隔符 构建hive表的完整demo

    1. add jar hdfs://team/work/libs/xxx/hive-contrib-2.0.1.jar;
    2. drop table if exists mb_tmp.yz_1013;
    3. create table if not exists mb_tmp.yz_1013(
    4. test_id int,
    5. work_place string,
    6. other string
    7. )
    8. partitioned by(dt string)
    9. ROW FORMAT SERDE 'org.apache.hadoop.hive.contrib.serde2.MultiDelimitSerDe' WITH SERDEPROPERTIES ("field.delim"="===")
    10. load data local inpath '/data/home/xxxx/client/ct_data_1010.txt' overwrite into table mb_tmp.yz_1013 partition(dt='2022-10-10');

    其中 以三个等号分割进行作为列分割的数据如下:

    1. [work@xxxx client]$ cat ct_data_1010.txt
    2. 10011===beijing===chu==shi
    3. 10012===tianjin===chang==chun
    4. 10013===zhengzhou===si==ji
    5. 100014===zhangda===j==ob

    4.3、结果验证

    1. hive> desc mb_tmp.yz_1013;
    2. OK
    3. test_id int from deserializer
    4. work_place string from deserializer
    5. other string from deserializer
    6. dt string
    7. # Partition Information
    8. # col_name data_type comment
    9. dt string
    10. Time taken: 0.332 seconds, Fetched: 9 row(s)
    11. hive> set hive.cli.print.header=true;
    12. hive> select * from mb_tmp.yz_1013 limit 100;
    13. OK
    14. yz_1013.test_id yz_1013.work_place yz_1013.other yz_1013.dt
    15. 10011 beijing chu==shi 2022-10-10
    16. 10012 tianjin chang==chun 2022-10-10
    17. 10013 zhengzhou si==ji 2022-10-10
    18. 100014 zhangda j==ob 2022-10-10
    19. Time taken: 0.51 seconds, Fetched: 4 row(s)
    20. hive> select other,'****',dt,work_place,test_id from mb_tmp.yz_1013 limit 100;
    21. OK
    22. other c1 dt work_place test_id
    23. chu==shi **** 2022-10-10 beijing 10011
    24. chang==chun **** 2022-10-10 tianjin 10012
    25. si==ji **** 2022-10-10 zhengzhou 10013
    26. j==ob **** 2022-10-10 zhangda 100014
    27. Time taken: 0.356 seconds, Fetched: 4 row(s)

    5、参考:

    1)、Hive多字符分隔符支持https://baijiahao.baidu.com/s?id=1617938645018071295&wfr=spider&for=pc

    2、Hive在0.14及以后版本支持字段的多分隔符,官方文档https://cwiki.apache.org/confluence/display/Hive/MultiDelimitSerDe

    3、Hive中的自定义分隔符(包含Hadoop和Hive详细安装)

     https://blog.csdn.net/github_39577257/article/details/89020980

    6、Hive 自带的多字符分割使用demo2

    默认情况下,Hive对于分隔符只支持单字符,不过Hive自带一个工具jar包,这个包支持正则和多字符方式定义分隔符。

    1). 查询hive自带的工具jar包位置
    find / -name hive-contrib-*.jar
    2). 将上面搜索到的jar包配置到配置hive-site.xml文件中

      hive.aux.jars.path
      file:///opt/apache-hive-1.2.2-bin/lib/hive-contrib-1.2.2.jar
      Added by tiger.zeng on 20120202.These JAR file are available to all users for all jobs

    上面配置之后可以不用重启Hive服务,只需要重新进入Hive CLI就可生效,且是永久的。也可以配置为临时的,就是在进入Hive CLI后,临时加载这个jar包,执行如下:
    hive> add jar file:///opt/apache-hive-1.2.2-bin/lib/hive-contrib-1.2.2.jar
    3). 使用
    准备如下数据,分隔符为 |#|,

    3324|#|003|#|20190816 09:16:18|#|0.00|#|2017-11-13 12:00:00
    3330|#|009|#|20190817 15:21:03|#|1234.56|#|2017-11-14 12:01:00

    建表时如下声明与定义如下,并加载数据,查询数据:

    drop table if exists split_test;
    CREATE  TABLE  split_test(
    id   INT COMMENT '借阅查询ID',
    number   STRING COMMENT '流水号',
    `date`   STRING COMMENT '查询返回日期',
    loanamount   DOUBLE COMMENT '借款金额范围',
    createtime   TIMESTAMP COMMENT '创建时间'
    )ROW FORMAT SERDE 'org.apache.hadoop.hive.contrib.serde2.MultiDelimitSerDe'
    WITH SERDEPROPERTIES ("field.delim"="|#|")
    STORED AS TEXTFILE;


     --加载数据
    LOAD DATA LOCAL INPATH '/root/split_test.txt'  OVERWRITE INTO TABLE split_test;
     
    查询结果如下:
    --查询数据
    hive> select * from split_test;
    OK
    3324    003     20190816 09:16:18       0.0     2017-11-13 12:00:00
    3330    009     20190817 15:21:03       1234.56 2017-11-14 12:01:00
    Time taken: 0.11 seconds, Fetched: 2 row(s)
     

    7、Hive 自定义分隔符例子


    自定义部分使用 Java 编写,使用Idea新建一个 Maven项目。
    https://blog.csdn.net/github_39577257/article/details/89020980

    8、hive特殊分隔符,不可见的分隔符 枚举

    1.hive默认字段分隔符^A(使用vim文本编辑显示),常用ASCII八进制 '\001' 或者UNICODE编码十六进制 '\u0001' 进行设置;通过notepad++打开显示为SOH

    2.hive特殊分隔符^B(使用vim文本编辑显示),常用ASCII八进制 '\002' 或者UNICODE编码十六进制 '\u0002' 进行设置;通过notepad++打开显示为STX

    3.hive特殊分隔符^C(使用vim文本编辑显示),常用ASCII八进制 '\003' 或者UNICODE编码十六进制 '\u0003' 进行设置;通过notepad++打开显示为ETX·

    此外,不同编程语音使用hive中\u0001、\0001、\0010、\u0010等分隔符切割需要不同写法

    以‘\u0010’为例
    hive sql使用split函数时,如果字段内使用的是特殊分隔符()需要用split(xxx,‘\u0010’)才能正常切割。
    如果是使用java写mr代码,那么需要使用的是"\0010"

    ​ 9、hive分隔符_HIVE-默认分隔符的(linux系统的特殊字符)查看,输入和修改

    hive表分隔符修改实操(兼容sqoop)_hive修改分隔符-CSDN博客

    #修改分隔符为逗号 ,

    ALTER TABLE table_name SET SERDEPROPERTIES ('field.delim' = ',' , 'serialization.format'=',');

    #修改分隔符为\001,在linux的vim中显示为^A,是hive默认的分隔符

    ALTER TABLE table_name SET SERDEPROPERTIES ('field.delim' = '\001' , 'serialization.format'='\001');

    #修改分隔符为制表符\t

    ALTER TABLE table_name SET SERDEPROPERTIES ('field.delim' = '\t' , 'serialization.format'='\t');


    重点知识:
    field.delim 指定表的两个列字段之间的文件中的字段分隔符.
    serialization.format 指定数据文件序列化时表中两个列字段之间的文件中的字段分隔符.
    对于分区表,每个分区可以有不同的分隔符属性
    alter语法修改分区表的分隔符后,不会影响已有分区数据读写,只会对后续新写入的数据生效。这一点非常友好
    alter语法修改分隔符只针对于后续新增数据有效,拿分区表而言,比如现在有2个分区,day=2020-05-01,day=2020-05-02,分隔符是\t, 通过alter把分隔符改为\001,再写入写的分区day=2020-05-03

    可以通过desc formatted tablename partition(key=value)语法查看每个分区的分隔符,那么2020-05-01,2020-05-02的分区依然是\t分隔符,2020-05-03分区的分隔符是\001;而且可以通过hive正常读写操作这三个分区而不会出现任何问题

    通过desc formatted table查看该表的分隔符,发现已经变为\001

    10、hive列分隔符和行分隔符概述

    一)、Hive中默认的分割符如下

    分隔符描述
    \n对于文本文件来说,每行都是一条记录,因此换行符可以分隔记录
    ^A(Ctrl+A)用于分隔字段(列)。在CREATE TABLE语句中可以使用八进制编码\001表示
    ^B用于分隔ARRAY或者STRUCT中的元素,或用于MAP中键-值对之间的分隔。在CREATE TABLE语句中可以使用八进制编码\002表示
    ^C用于MAP中键和值之间的分隔。在CREATE TABLE语句中可以使用八进制编码\003表示


     二)、分隔符的指定与使用
    hive中在创建表时,一般会根据导入的数据格式来指定字段分隔符和列分隔符。一般导入的文本数据字段分隔符多为逗号分隔符或者制表符(但是实际开发中一般不用着这种容易在文本内容中出现的的符号作为分隔符),当然也有一些别的分隔符,也可以自定义分隔符。有时候也会使用hive默认的分隔符来存储数据。
     

    1. hive (fdm_sor)> create table fdm_sor.mytest_tmp2(
    2. > id int comment'编号',
    3. > name string comment '名字'
    4. > );
    5. hive (fdm_sor)> show create table mytest_tmp2;
    6. CREATE TABLE `mytest_tmp2`(
    7. `id` int COMMENT '编号',
    8. `name` string COMMENT '名字')
    9. ROW FORMAT SERDE
    10. 'org.apache.hadoop.hive.serde2.lazy.LazySimpleSerDe' --hive默认的分割方式,即行为\n,列为^A
    11. STORED AS INPUTFORMAT
    12. 'org.apache.hadoop.mapred.TextInputFormat' --hive默认的存储格式为textfile
    13. OUTPUTFORMAT
    14. 'org.apache.hadoop.hive.ql.io.HiveIgnoreKeyTextOutputFormat'
    15. LOCATION --内部表的默认的存储路径
    16. 'hdfs://hadoop102:9000/user/hive/warehouse/fdm_sor.db/mytest_tmp2'
    17. TBLPROPERTIES (
    18. 'transient_lastDdlTime'='1526176805')
    19. hive (fdm_sor)> create table fdm_sor.mytest_tmp3(
    20. > id int comment'编号',
    21. > name string comment '名字'
    22. > )
    23. > row format delimited
    24. > fields terminated by '\001' --这里可以指定别的分隔符,如‘\t’,'$'等分隔符
    25. > COLLECTION ITEMS TERMINATED BY '\002' -- 集合间的分隔符
    26. > MAP KEYS TERMINATED BY '\003' -- Map键与值之间的分隔符
    27. > LINES TERMINATED BY '\n' -- 行分隔符
    28. > stored as textfile; -- 存储格式为textfile
    29. hive (fdm_sor)> show create table fdm_sor.mytest_tmp3;
    30. OK
    31. createtab_stmt
    32. CREATE TABLE `fdm_sor.mytest_tmp3`(
    33. `id` int COMMENT '编号',
    34. `name` string COMMENT '编号')
    35. ROW FORMAT DELIMITED
    36. FIELDS TERMINATED BY '\u0001'
    37. LINES TERMINATED BY '\n'
    38. STORED AS INPUTFORMAT
    39. 'org.apache.hadoop.mapred.TextInputFormat'
    40. OUTPUTFORMAT
    41. 'org.apache.hadoop.hive.ql.io.HiveIgnoreKeyTextOutputFormat'
    42. LOCATION
    43. 'hdfs://hadoop102:9000/user/hive/warehouse/fdm_sor.db/mytest_tmp3'
    44. TBLPROPERTIES (
    45. 'transient_lastDdlTime'='1526176859')

    -注意:ROW FORMAT DELIMITED这组关键字必须要写在其他子句(除了STORED AS…)子句之前。

     如上可以看出hive默认的列分割类型为org.apache.hadoop.hive.serde2.lazy.LazySimpleSerDe,

    而这其实就是^A分隔符,hive中默认使用^A(ctrl+A)作为列分割符,如果用户需要指定的话,等同于row format delimited fields terminated by '\001',因为^A八进制编码体现为'\001'.所以如果使用默认的分隔符,可以什么都不加,也可以按照上面的指定加‘\001’为列分隔符,效果一样。

           hive默认使用的行分隔符是'\n'分隔符 ,也可以加一句:LINES TERMINATED BY '\n' ,加不加效果一样。但是区别是hive可以通过row format delimited fields terminated by '\t'这个语句来指定不同的分隔符,但是hive不能够通过LINES TERMINATED BY '$$'来指定行分隔符,目前为止,hive的默认行分隔符仅支持‘\n’字符。否则报错。

    1. hive (fdm_sor)> create table fdm_sor.mytest_tm4(
    2. > id int comment'编号',
    3. > name string comment '名字'
    4. > )
    5. > lines terminated by '\t';
    6. FAILED: ParseException line 5:1 missing EOF at 'lines' near ')'

             一般来说hive的默认行分隔符都是换行符,如果非要自定义行分隔符的话,可以通过自定义Inputformat和outputformat类来指定特定行分隔符和列分隔符,一般公司实际开发中也都是这么干的,具体使用。

           当然如hive中集合数据类型struct ,map,array,也都有默认的字段分隔符,也都可以指定字段分隔符。hive中对于上述三个集合数据类型的默认字段分隔符是^B,八进制体现为‘\002’,用collection items terminated by '\002'语句来指定分隔符,对于map来说,还有键值之间的分割符,可以用map keys terminated by  '\003'(^C)来指定分隔符。

    三)、建好表之后更改字段分隔符

       参见第九步

  • 相关阅读:
    面试突击52:什么是三范式?它有什么用?
    R语言 | 安装ggpubr R包时编译语句中出现 WARNING: ignoring environment value of R_HOME 而报错
    2 路 2GSPS/2.6GSPS/3GSPS 14bit AD 采集卡/ FMC 子卡 (AD9208/ AD9689)
    云计算 3月15号(linux的权限管理 进程管理 管道重定向 磁盘管理 文件查找 软件管理 计划任务 web服务器 配置文件)
    面试官:介绍一下 Redis 三种集群模式
    linux上 more 和 cat 区别
    记 linux 系统编译好的exp提权提示无gcc
    设计模式:装饰器模式
    如何避免邮件被识别为垃圾邮件
    Java开发必须掌握的运维知识 (十)-- Docker集群自动化部署管理:Kubernetes快速入门
  • 原文地址:https://blog.csdn.net/helloxiaozhe/article/details/133748653