[Hive]Hive中表连接的优化,加快查询速度

1、多表连接的执行顺序和MapReduce job优化

select
    a.ymd
    ,a.price_close
    ,b.price_close
    ,c.price_close
from stocks a join stocks b on a.ymd = b.ymd
              join stocks c on a.ymd = c.ymd
where a.symbol = 'APPL' and b.symbol = 'IBM' and c.symbol = 'GE'

 

大多数情况下Hive会对每对 JOIN 连接对象启动一个MapReduce任务。上例中会首先启动一个MapReduce job对表 a 和表 b 进行连接操作,
然后会启动一个MapReuce job将第一个MapReduce job的输出和表 c 进行连接操作。

当对3个或者更多个表进行 JOIN 连接时,如果每个 ON 子句都使用相同的连接键的话,那么只会产生一个 Map Reduce job。
即Hive通过一个优化可以在同一个Map Reduce job 中连接多个表(需要每个子句中有相同的连接键)。

2、JOIN连接时表的顺序(内连接)
Hive假定查询中最后一个表是最大的那个表,在对每行记录进行连接操作时,它会尝试将其他表缓存起来,然后扫描最后那个表进行计算。
因此用户需要保证连续查询中表的大小从左往右是依次增加的。

用户也可以人为显示告诉查询优化器哪张表是大表,即使它在查询中不是位于最后面的,使用方式如下:

select /*+STREAMTABLE(s)*/s.ymd, s.symbol,s.price_close,d.dividend
from stocks s JOIN dividends d ON s.ymd = d.ymd and s.symbol = d.symbol
where s.symbol='APPL'

3、连接前通过WHERE过滤数据加快查询数据
最直接了当的方式是使用嵌套SELECT语句:

select 
    s.ymd,s.symbol,s.price_close,d.dividends 
from 
    (
        select * from stocks 
        where symbol = 'AAPL' and exchange = 'NASDAQ'
    )s
left outer join
    (
        select * from dividends 
        where symbol = 'AAPL' and exchange = 'NASDAQ'
    )d
on s.ymd = d.ymd

4、map-side JOIN 优化含有小表的连接
如果所有表中只有一张表是小表,则可以在最大的表通过mapper的时候将小表完全放到内存中。Hive可以在map端执行连接过程(称为mao-side JOIN),这是因为
Hive可以和内存中的小表逐一匹配,从而省略常规连接操作需要的reduce过程。不仅减少了reduce过程,有时也可以同时减少map过程的执行步骤。

set hive.auto.convert.join=true; --让Hive在必要的时候启动优化
hive.mapjoin.smalltable.filesize=25000000; --设置能够使用这个优化的小表的大小(单位是字节)

 

  • 1
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Hive中,如果要将一个大表与一个小表进行join,可以考虑以下两种方式: 1. Map-join:将小表加载到内存中,然后将大表与内存中的小表进行join。这种方式的优点是可以极大地提高join的效率,缺点是需要足够的内存来存放小表。 2. Broadcast-join:将小表直接广播到每个map任务中,然后将大表与每个map任务中的小表进行join。这种方式的优点是不需要过多的内存,适用于小表较小时的情况。缺点是需要额外的网络传输开销。 在Hive中,可以使用以下语句实现Map-join: ``` set hive.auto.convert.join=true; set hive.mapjoin.smalltable.filesize=25000000; //设置小表大小 SELECT /*+ MAPJOIN(smalltable) */ large_table.*, small_table.* FROM large_table JOIN small_table ON (large_table.key = small_table.key); ``` 可以使用以下语句实现Broadcast-join: ``` set hive.auto.convert.join=true; set hive.auto.convert.join.noconditionaltask=true; SELECT /*+ BROADCASTJOIN(smalltable) */ large_table.*, small_table.* FROM large_table JOIN small_table ON (large_table.key = small_table.key); ``` ### 回答2: 在Hive中,大表与小表的联接操作需要进行一些特别的考虑。首先,大表与小表的大小差异会影响到性能和效率。由于大表通常包含大量的数据,而小表相对较小,因此,联接操作时需要将小表加载到内存中,以避免频繁的磁盘读取。这可以通过将小表设置为Hive的Map Join进行优化。 Map Join是一种在内存中进行的联接操作,它将小表缓存到节点的内存中,并在联接时直接在内存中进行操作,从而大大减少了磁盘I/O操作,提高了性能。要使用Map Join,可以使用HiveQL中的"mapjoin"关键字来启用此功能。 但是,应注意的是,由于内存有限,只有小表可以完全加载到内存中,因此只有当小表的大小适合全部加载到内存中时才适合使用Map Join。大表则没有这个限制。 此外,在进行大表与小表的联接操作时,还可以考虑使用特定的联接算法来提高性能,例如,使用哈希联接(Hash Join)来减少I/O操作,或者使用排序联接(Sort Merge Join)来加快排序操作。 最后,对于Hive大表与小表的联接操作,还可以通过对数据进行分区或者使用分桶技术来提高性能。通过分区将大表和小表划分为更小的块,可以减少联接操作需要处理的数据量。而分桶则可以根据某个列的哈希值将数据分布到不同的桶中,从而更高效地进行联接操作。 总结起来,Hive中大表与小表的联接操作需要考虑优化性能和效率的因素,例如使用Map Join、选择适当的联接算法、采用分区或分桶技术等。 ### 回答3: 在Hive中,进行大表和小表的连接操作通常有两种方式:MapJoin(映射连接)和ReduceJoin(归并连接)。 1. MapJoin:当小表可以全部存放在内存中时,使用MapJoin可以将小表加载到内存中,并将其构建为一个哈希表。随后,对大表进行扫描,将符合连接条件的记录通过哈希表进行匹配,以获取结果。MapJoin的优点是速度快、简单,并且节省了磁盘IO的开销。然而,由于小表需要完全加载到内存中,可能会导致内存资源紧张,且对于大型数据集来说可能并不适用。 2. ReduceJoin:当无法将小表全部加载到内存中时,就需要使用ReduceJoin。ReduceJoin将大表和小表根据连接字段进行排序,然后按照连接字段进行分区。在后续的reduce阶段,将同一个连接字段的记录发送到同一个reduce任务中,然后通过遍历大表和小表的方式进行匹配,最终得到结果。ReduceJoin的优点是适用于大型数据集,并且不需要将整个小表加载到内存中,避免了内存资源的浪费。然而,ReduceJoin的缺点是耗时较长,因为需要进行排序和分区操作,并且需要额外的磁盘IO。 总之,对于Hive中的大表和小表连接操作,根据实际情况选择合适的连接方式,可以提高查询的效率和性能。如果小表可以全部加载到内存中,使用MapJoin可以获得更快的结果;如果小表不能完全加载到内存中,则可以使用ReduceJoin来处理大规模数据集。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值