赞
踩
本篇文章主要整理hive-3.1.2版本的企业调优经验,有误请指出~
使用explain命令可以分析查询计划,查看计划中的资源消耗情况,定位潜在的性能问题,并进行相应的优化。
explain执行计划见文章:
根据集群的配置和资源情况,合理调整Hive查询的并行度和资源分配,可以提高查询的并发性和整体性能。Hive在实现HQL计算运行时,会解析为多个Stage,有时候Stage彼此之间有依赖关系,只能挨个执行,但是在一些别的场景下,很多的Stage之间是没有依赖关系的。
例如Union语句,Join语句等等,这些Stage没有依赖关系,但是Hive依旧默认挨个执行每个Stage,这样会导致性能非常差,我们可以通过修改参数,开启并行执行,当多个Stage之间没有依赖关系时,允许多个Stage并行执行,提高性能。
- -- 开启Stage并行化,默认为false
- SET hive.exec.parallel=true;
- -- 指定并行化线程数,默认为8
- SET hive.exec.parallel.thread.number=16;
ps:调整并行度的措施,建议在数据量大,sql 逻辑复杂的时候使用。当数据量小或sql逻辑简单时开启并行度,优化效果不明显。
使用hive的过程中,有一些数据量不大的表也会抓换成MapReduce处理,提交到yarn集群时,需要申请资源,等待资源分配,启动JVM进行,再运行Task,一系列的过程比较繁琐,严重影响性能。Hive为解决这个问题,延用了MapReduce中的设计,提供本地计算模式,允许程序不提交给yarn,直接在本地运行。
- -- 开启本地模式
- set hive.exec.mode.local.auto = true;
Fetch 抓取是指:Hive 中对某些简单的查询可以不必使用 MapReduce 计算。 hive-default.xml.template 配置文件的 hive.fetch.task.conversion 默认是 more。此时在全局查找、字段查找、limit 查找等都不走mapreduce。例如:select * from employees;
- 分区表、分桶表不是建表的必要语法规则,是一种优化手段表,可选;
- 分区好处:用where进行分区过滤,查询指定分区的数据,避免全表扫描
- 分桶好处:基于分桶字段查询时,减少全表扫描;join时可以转换为SMB(Sort Merge Bucket join)
- 分区针对的是数据的存储路径;分桶针对的是数据文件(数据粒度更细)
- 分区字段不能是表中已经存在的字段,分桶的字段必须是表中已经存在的字段
- 分区字段是虚拟字段,其数据并不存储在底层的文件中;
- 分区字段值可以手动指定(静态分区),也可以根据查询结果位置自动推断(动态分区)
- Hive支持多重分区,也就是说在分区的基础上继续分区,支持更细粒度的目录划分
详细信息见文章:
Hive Join的底层是通过MapReduce来实现的,Hive实现Join时,为了提高MapReduce任务的性能,提供了多种Join方案来实现。例如:适合小表Join大表的Map Join,大表Join大表的Reduce Join, 以及大表Join的优化方案Bucket Join等。
1)应用场景:小表join大表、小表Join小表
2)概述:Map Join是直接在Map阶段完成join工作,没有Shuffle阶段,从而避免了数据倾斜。
- select /*+ mapjoin(b,c)*/ --mapjoin hint 定义小表,多个小表用逗号分隔
- ...
- from t0 a
- left join t1 b
- on a.id = b.id
- left join t2 c
- on a.id = c.id;
-
- # MapJoin中多个小表用半角逗号(,)分隔,例如/*+ mapjoin(a,b,c)*/。
3)工作机制:使用hadoop中DistributedCache(分布式缓存)将小表广播到每个map任务节点,转换成哈希表加载到内存中,之后在mapper端和大表的分散数据做笛卡尔积,直接输出结果。
4)Map Join的特点:
- 要使用hadoop中的DistributedCache(分布式缓存)把小数据分布到各个计算节点,每个map节点都要把小数据库加载到内存,按关键字建立索引。
- Map Join没有reduce任务,所以map直接输出结果,即有多少个map任务就会产生多少个结果文件
- Hive3.1.2版本已经对Map Join进行了优化,小表放在左边和右边已经没有区别
- MapJoin在Map阶段会将指定表的数据全部加载在内存中,因此指定的表仅能为小表且表被加载到内存后占用的总内存不得超过512 MB(默认)。由于MaxCompute是压缩存储,因此小表在被加载到内存后,数据大小会急剧膨胀。
5)参数设置:
- #设置自动选择 Mapjoin,默认为true
- set hive.auto.convert.join = true;
- #大表小表的阈值设置(默认25M以下认为是小表)
- set hive.mapjoin.smalltable.filesize = 25*1000*1000;
1)应用场景:大表Join大表
2)概述:将两张表按照相同的规则将数据划分、根据对应的规则的数据进行join、减少了比较次数,提高了性能
Sort Merge Bucket Join:基于有序的数据Join
- set hive.optimize.bucketmapjoin = true;
- set hive.auto.convert.sortmerge.join=true;
- set hive.optimize.bucketmapjoin.sortedmerge = true;
- set hive.auto.convert.sortmerge.join.noconditionaltask=true;
- # 创建分桶表 bigtable_buck1
- create table bigtable_buck1(
- id bigint,
- t bigint,
- uid string,
- keyword string,
- url_rank int,
- click_num int,
- click_url string
- )
- clustered by(id)
- sorted by(id)
- into 6 buckets
- row format delimited fields terminated by '\t';
-
- # 加载数据
- load data local inpath '/opt/module/data/bigtable' into table
- bigtable_buck1;
-
-
- #创建分桶表bigtable_buck2,分桶数和bigtable_buck1的分桶数为倍数关系
- create table bigtable_buck2(
- id bigint,
- t bigint,
- uid string,
- keyword string,
- url_rank int,
- click_num int,
- click_url string
- )
- clustered by(id)
- sorted by(id)
- into 6 buckets
- row format delimited fields terminated by '\t';
-
- #加载数据
- load data local inpath '/opt/module/data/bigtable' into table
- bigtable_buck2;
-
- #================ SMB Join调优步骤
-
- #设置参数
- set hive.optimize.bucketmapjoin = true;
- set hive.optimize.bucketmapjoin.sortedmerge = true;
- set hive.input.format=org.apache.hadoop.hive.ql.io.BucketizedHiveInputFormat;
-
-
- # SMB Join
- insert overwrite table jointable
- select b.id,
- b.t,
- b.uid,
- b.keyword,
- b.url_rank,
- b.click_num,
- b.click_url
- from bigtable_buck1 s
- join bigtable_buck2 b
- on b.id = s.id;
1)应用场景:大表Join大表
2)概述:Skew Join是Hive中一种专门为了避免数据倾斜而设计的特殊的Join过程。这种Join的原理是将Map Join和Reduce Join进行合并,如果某个值出现了数据倾斜,就会将产生数据倾斜的数据单独使用Map Join来实现其他没有产生数据倾斜的数据由Reduce Join来实现,这样就避免Reduce Join中产生数据倾斜的问题,最终将Map Join的结果和Reduce Join的结果进行Union合并
3)SkewJoin Hint 语法
select /*+ skewJoin(<table_name>[(<column1_name>[,<column2_name>,...])][((<value11>,<value12>)[,(<value21>,<value22>)...])]*/
在select
语句中使用上述的Hint提示才会执行MapJoin,其中table_name
为倾斜表名,column_name
为倾斜列名,value
为倾斜Key值。使用示例如下:
- #=========性能效率 方法1 < 方法2 <方法3
- #=========下面三个级别的hint,指定热点信息越详细,效率越高。
-
- #-- 方法1:Hint表名(注意Hint的是表的别名)。
- select /*+ skewjoin(a)*/ ... from t0 a join t1 b on a.id = b.id and a.code = b.code;
-
- #-- 方法2:Hint表名和认为可能产生倾斜的列,下面认为a表的id和code列 存在倾斜
- select /*+ skewjoin(a(id,code))*/ ... from t0 a join t1 b on a.id = b.id and a.code = b.code;
-
- #-- 方法3:Hint表名和列,并提供产生倾斜的列的值,如果是string类型需要加引号,
- #--此案例 认为 (a.id=1 and a.code='xxx') 和 (a.id=3 and a.code='yyy') 的值出现倾斜
- select /*+ skewjoin(a(id,code)((1,'xxx'),(3,'yyy')))*/ ... from t0 a join t1 b on a.id = b.id and a.code = b.code;
4)Skew Join参数:
- #-- 开启运行过程中skewjoin
- set hive.optimize.skewjoin=true;
- #-- 如果这个key的出现的次数超过这个范围
- set hive.skewjoin.key=100000;
- #-- 在编译时判断是否会产生数据倾斜
- set hive.optimize.skewjoin.compiletime=true;
- #-- 不合并,提升性能
- set hive.optimize.union.remove=true;
- #-- 如果Hive的底层走的是MapReduce,必须开启这个属性,才能实现不合并
- set mapreduce.input.fileinputformat.input.dir.recursive=true;
1)应用场景:大表Join大表
2)概述:两张表的数据关联会经过shuffle阶段,Hive会自动判断是否满足Map Join,如果不满足Map Join,则自动执行Reduce Join(普通的join方式)
3)阶段阐述:
生成键值对,以join on 条件中的列作为key,以join之后所关心的列作为value值,在value中还会包含Tag标记信息,用于标明此value对应哪张表
根据key值进行hash分区, 按照hash值将键值对(key-value)发送到不同的reducer中
Reducer通过Tag来识别不同的表中的数据,然后分别做合并操作
4)sql举例:
- SELECT pageid,
- age
- FROM page_view
- JOIN userinfo
- ON page_view.userid = userinfo.userid;
sql转化为mr任务流程如下图:
5)Reduce Join方法缺点:
- map阶段没有对数据瘦身,shuffle的网络传输和排序性能很低。
- reduce端需要通过Tag识别来源不同表的数据,很耗内存,容易导致OOM。
如果分组字段本身存在大量重复值(该字段值也叫做热点key值),group by底层走shuffle,分组聚合会出现数据倾斜的现象,可以使用如下三种方案解决:
序号 | 方案 | 说明 |
方案一 | 开启Map端聚合 | set hive.map.aggr=true; |
方案二 | 数据倾斜时自动负载均衡 | set hive.groupby.skewindata = true; |
方案三 | 添加随机数,两阶段聚合 | 热点key加随机数,阶段拆分,两阶段聚合 |
- #--开启Map端聚合,默认为true
- set hive.map.aggr = true;
- #--在Map 端预先聚合操作的条数
- set hive.groupby.mapaggr.checkinterval = 100000;
该参数可以将顶层的聚合操作放在 Map 阶段执行,从而减轻shuffle清洗阶段的数据传输和 Reduce阶段的执行时间,提升总体性能
- #---有数据倾斜的时候自动负载均衡(默认是 false)
- set hive.groupby.skewindata = true;
开启该参数以后,当前程序会自动通过两个MapReduce来运行
该参数的优化原理是:将M->R阶段 拆解成 M->R->R阶段
引入两级reduce:
第一阶段的shuffle key = group key + 随机数,将热点数据打散,多个reduce并发做部分聚合;
第二阶段的shuffle key = group key,保障相同key分发到同一个reduce做最终聚合;
-
- #===============优化前
- insert overwrite table tblB partition (dt = '2022-10-19')
- select
- cookie_id,
- event_query,
- count(*) as cnt
- from tblA
- where dt >= '20220718'
- and dt <= '20221019'
- and event_query is not null
- group by cookie_id, event_query
-
-
-
- #===============优化后
- insert overwrite table tblB partition (dt = '2022-10-19')
- select
- split(tkey, '_')[1] as cookie_id,
- event_query,
- #--- 第二阶段2:求出最终的聚合值
- sum(cnt) as cnt
- from (
- select
- concat_ws('_', cast(ceiling(rand() * 99) as string), cookie_id) as tkey,
- event_query,
- #---第一阶段2:先局部聚合得到cnt
- count(*) as cnt
- from tblA
- where dt >= '20220718'
- and dt <= '20221019'
- and event_query is not null
- #--- 第一阶段1:添加[0-99]随机整数,将热点Key值:cookie_id 进行打散( M -->R)
- group by concat_ws('_', cast(ceiling(rand() * 99) as string), cookie_id),
- event_query
- ) temp
- #--- 第二阶段1:对拼接的key值进行切分,还原原本的key值split(tkey, '_')[1] =cookie_id ( R -->R)
- group by split(tkey, '_')[1], event_query;
优化思路为:
count (distinct) 使得map端无法预聚合,容易引发reduce端长尾。数据量大可以使用group by替换count ..distinct去实现去重,但是需要注意:如果count(distinct 大量重复数据),group by可能会带来数据倾斜问题(对热点key分组操作,容易导致数据倾斜)。
用group by 替换 count(distinct)的案例见文章:
- RBO(rule basic optimise)基于规则的优化器,根据设定好的规则来对程序进行优化;
- CBO(cost basic optimise)基于代价的优化器,根据不同场景所需要付出的代价来合适选择优化的方案对数据的分布的信息【数值出现的次数,条数,分布】来综合判断用哪种处理的方案是最佳方案;Hive中支持RBO与CBO这两种引擎,默认使用的是RBO优化器引擎。
根据不同的应用场景,可以选择CBO,设置方式如下:
- set hive.cbo.enable=true;
- set hive.compute.query.using.stats=true;
- set hive.stats.fetch.column.stats=true;
- set hive.stats.fetch.partition.stats=true;
用于提前运行一个MapReduce程序,基于表或者分区的信息构建元数据信息【包含表的信息、分区信息、列的信息】,搭配CBO引擎一起使用。
- -- 构建分区信息元数据
- ANALYZE TABLE tablename
- [PARTITION(partcol1[=val1], partcol2[=val2], ...)]
- COMPUTE STATISTICS [noscan];
-
- -- 构建列的元数据
- ANALYZE TABLE tablename
- [PARTITION(partcol1[=val1], partcol2[=val2], ...)]
- COMPUTE STATISTICS FOR COLUMNS ( columns name1, columns name2...) [noscan];
-
- -- 查看元数据
- DESC FORMATTED [tablename] [columnname];
-
- --分析优化器
- --构建表中分区数据的元数据信息
- ANALYZE TABLE tb_login_part PARTITION(logindate) COMPUTE STATISTICS;
- --构建表中列的数据的元数据信息
- ANALYZE TABLE tb_login_part COMPUTE STATISTICS FOR COLUMNS userid;
- --查看构建的列的元数据
- desc formatted tb_login_part userid;
谓词下推Predicate Pushdown(PPD):在不影响结果的情况下,尽量将过滤条件提前执行。谓词下推后,过滤条件在map端执行,减少了map端的输出,降低了数据在集群上传输的量,提升任务性能。
谓词下推的场景分析见文章:
in/exists操作,推荐使用Hive的left semi join(左半连接)进行替代
- # in / exists 实现
- select a.id, a.name from a where a.id in (select b.id from b);
- select a.id, a.name from a where exists (select id from b where a.id =
- b.id);
-
- #可以使用 join 来改写
- select a.id, a.name from a join b on a.id = b.id;
-
- #left semi join 实现
- select a.id, a.name from a left semi join b on a.id = b.id;
left semi join(左半连接)的详细说明见文章:
hive/spark--left semi/anti join_sparksql left semi join-CSDN博客
拖慢HQL查询效率的原因除了join引发的shuffle过程外,还有一个就是子查询调用次数较多,存在冗余代码块。因此我们可以借助CTE 公共表达式,简单来讲就是:with as 语句,将代码中的子查询事先提取出来(类似临时表),可以避免重复计算。
- #==============优化前
- select
- uid,
- --每个用户一月份的订单数
- sum(if(dt = '2018-01', 1, 0)) as m1_count,
- --每个用户二月份的订单数
- sum(if(dt = '2018-02', 1, 0)) as m2_count,
- --每个用户三月份的订单数(当月订单金额超过10元的订单个数)
- sum(if(dt = '2018-03' and oamount > 10, 1, 0)) m3_count,
- --当月(3月份)首次下单的金额
- sum(if(dt = '2018-03' and rk = 1, oamount, 0)) m3_first_amount,
- --当月(3月份)末次下单的金额(rk =cnt小技巧)
- sum(if(dt = '2018-03' and rk = cnt, oamount, 0)) m3_last_amount
- from (
- select
- oid,
- uid,
- otime,
- date_format(otime, 'yyyy-MM') as dt,
- oamount,
- ---计算rk的目的是为了获取记录中的第一条
- row_number() over (partition by uid,date_format(otime, 'yyyy-MM') order by otime) rk,
- --- 计算cnt的目的是为了获取记录中的最后一条
- count(*) over (partition by uid,date_format(otime, 'yyyy-MM')) cnt
- from t_order
- order by uid
- ) tmp
- group by uid
- having m1_count > 0
- and m2_count = 0;
-
-
- #================优化后
- with tmp as (
- select
- oid,
- uid,
- otime,
- date_format(otime, 'yyyy-MM') as dt,
- oamount,
- ---计算rk的目的是为了获取记录中的第一条
- row_number() over (partition by uid,date_format(otime, 'yyyy-MM') order by otime) rk,
- --- 计算cnt的目的是为了获取记录中的最后一条
- count(*) over (partition by uid,date_format(otime, 'yyyy-MM')) cnt
- from t_order
- order by uid
- )
- select
- uid,
- --每个用户一月份的订单数
- sum(if(dt = '2018-01', 1, 0)) as m1_count,
- --每个用户二月份的订单数
- sum(if(dt = '2018-02', 1, 0)) as m2_count,
- --每个用户三月份的订单数(当月订单金额超过10元的订单个数)
- sum(if(dt = '2018-03' and oamount > 10, 1, 0)) m3_count,
- --当月(3月份)首次下单的金额
- sum(if(dt = '2018-03' and rk = 1, oamount, 0)) m3_first_amount,
- --当月(3月份)末次下单的金额(rk =cnt小技巧)
- sum(if(dt = '2018-03' and rk = cnt, oamount, 0)) m3_last_amount
- from tmp
- group by uid
- having m1_count >0 and m2_count=0;
ps:hive中的CTE公共表达式文章:
1)通常情况下,作业会通过input的目录产生一个或者多个map 任务,map数量主要的决定因素有:input 的文件总个数,input 的文件大小,集群设置的文件块大小。
当input的文件很大,任务逻辑复杂,map执行非常慢的时候,可以考虑增加Map数,公式:
computeSliteSize(Math.max(minSize,Math.min(maxSize,blocksize)))
让maxSize低于blocksize,此时公式 computeSliteSize(Math.max(minSize,Math.min(maxSize,blocksize))) = maxSize
原map个数= 输入文件的数据量 / computeSliteSize = 输入文件的数据量 / blocksize, 现在map个数=输入文件的数据量 / computeSliteSize = 输入文件的数据量 / maxSize
- # 每个reduce任务处理的数据量默认是 256MB
- hive.exec.reducers.bytes.per.reducer=256*1000*1000
- # 整个MR任务支持开启的reduce数的上限值,默认为1009个
- hive.exec.reducers.max=1009
根据该公式可以得知:reduce个数(hdfs上的落地文件数量)是动态计算的
在hadoop的mapred-default.xml 文件中修改下列参数,
- #设置MR job的Reduce总个数
- set mapreduce.job.reduces = 15;
ps:reduce个数并不是越多越好
Hive小文件问题及解决方案,见文章:
Hive中数据倾斜的解决方案指南见:
参考文章:
大数据从业者必知必会的Hive SQL调优技巧 | 京东云技术团队 - 掘金
https://zhugezifang.blog.csdn.net/article/details/127447167
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。