Hive基础三(查询中常用的语法)

相关链接:
Hive基础一(数据库,表,分区表,视图,导入导出数据)
Hive基础二(join原理和机制,join的几种类型,数据倾斜简单处理)
Hive基础三(查询中常用的语法)

Hive中的SELECT基础语法和标准SQL语法基本一致,支持WHERE、DISTINCT、GROUP BY、ORDER BY、HAVING、LIMIT、子查询等。

一,ORDER BY和SORT BY

ORDER BY根据key进行全局排序,由一个Reduce Task来完成。
SORT BY用于分区内排序,即有多个reduce,在每个Reduce任务内排序。
看下面的例子,原表数据为:

hive> select * from lxw1234_com;
OK
5
3
6
2
9
8
1

使用ORDER BY:

hive> select * from lxw1234_com order by id;
1
2
3
5
6
8
9

使用SORT BY:

hive> set mapred.reduce.tasks=2;
hive> select * from lxw1234_com sort by id;
2
5
6
9
1
3
8

设定了2个reduce,从结果可以看出,每个reduce内做了排序。如果reduce数为1,那么ORDER BY和SORT BY的结果是一样的。真实业务环境中,我们的需求大多需要使用ORDER BY全局排序来完成,默认是升序,降序是字段后加desc

二,DISTRIBUTE BY和CLUSTER BY

distribute by:按照指定的字段或表达式对数据进行划分,输出到对应的Reduce或者文件中,reduce数量大于1才能有效果。
cluster by:除了兼具distribute by的功能,还兼具sort by的排序功能。
使用distribute by:

hive> set mapred.reduce.tasks=2;
hive>INSERT overwrite LOCAL directory '/tmp/lxw1234/' 
SELECT id FROM lxw1234_com 
distribute BY id;

执行后在本地的/tmp/lxw1234目录中生成了000000_0和000001_0两个文件,是按照奇偶划分的(如果是按照length(id)划分,则长度相同的会划分在一个reduce):

$cat 000000_0
8
2
6
$cat 000001_0 
1
9
3
5

使用cluster by:

hive> set mapred.reduce.tasks=2;
hive> INSERT overwrite LOCAL directory '/tmp/lxw1234/' 
SELECT id FROM lxw1234_com 
CLUSTER BY id;
$cat 000000_0
2
6
8
$cat 000001_0
1
3
5
9

注意:使用cluster by之后,每个文件中的id都进行了排序,而distribute by没有。

三,group by的实现原理

以下面的语句为例,以省市为关键字进行分组:

select province,city,count(*) from table1 group by province,city;

思路:既然以省和市作为group by的字段,那就把他们的组合作为map的输出key,输出value置为1,经过shuffle之后,相同的省市组合会被合并,然后进入reduce,拆开省市组合作为两个字段,累加value作为count。
从Hive0.7.0开始HAVING被添加到Hive作为GROUP BY结果集的条件过滤。

四,distinct的实现原理

以下面的语句为例,以省为关键字进行分组,计算去重后的市的数量:

select province, count(distinct city) from table1 group by province;

思路:当只有一个distinct字段时,只需要将Group By字段和Distinct字段组合为map的输出key,value随意,经过shuffle之后,相同key合并,此时便完成了distinct操作(因为合并之后,相同的省后边的市肯定不同),然后先进入combine,把key拆开(省作为新的key,市作为新的value),再经过一次shuffle,相同的省合并,不同的市组成value-list,最后进入reduce即可。
如果是多个distinct呢?参看:
https://tech.meituan.com/hive-sql-to-mapreduce.html

五,UNION ALL和子查询

对两个查询结果进行并集操作,包括重复行,不进行排序,两个查询结果的字段必须一致。
下面这个例子还用了一种将子查询作为一个表的语法,叫做Common Table Expression(CTE):

with q1 as (select * from src where key= '5'),
q2 as (select * from src s2 where key = '4')
select * from q1 union all select * from q2;

子查询和标准SQL中的子查询语法和用法基本一致,需要注意的是,Hive中如果是从一个子查询进行SELECT查询,那么子查询必须设置一个别名:

SELECT col FROM (
SELECT a+b AS col
FROM t1
) t2

另外,从Hive0.13开始,在WHERE子句中也支持子查询,比如:

SELECT * FROM A
WHERE A.a IN (SELECT foo FROM B);
-----
SELECT A FROM T1
WHERE EXISTS (SELECT B FROM T2 WHERE T1.X = T2.Y)


原文来自:http://lxw1234.com/archives/2015/07/365.htm

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值