当前位置:   article > 正文

Hive常用函数大全(二)(窗口函数、分析函数、增强group)_hive last value聚合函数

hive last value聚合函数

窗口函数与分析函数

应用场景: 
(1)用于分区排序 
(2)动态Group By 
(3)Top N 
(4)累计计算 
(5)层次查询

窗口函数

FIRST_VALUE:取分组内排序后,截止到当前行,第一个值 
LAST_VALUE: 取分组内排序后,截止到当前行,最后一个值 
LEAD(col,n,DEFAULT) :用于统计窗口内往下第n行值。第一个参数为列名,第二个参数为往下第n行(可选,默认为1),第三个参数为默认值(当往下第n行为NULL时候,取默认值,如不指定,则为NULL) 
LAG(col,n,DEFAULT) :与lead相反,用于统计窗口内往上第n行值。第一个参数为列名,第二个参数为往上第n行(可选,默认为1),第三个参数为默认值(当往上第n行为NULL时候,取默认值,如不指定,则为NULL)

OVER从句

1、使用标准的聚合函数COUNT、SUM、MIN、MAX、AVG 
2、使用PARTITION BY语句,使用一个或者多个原始数据类型的列 
3、使用PARTITION BY与ORDER BY语句,使用一个或者多个数据类型的分区或者排序列 
4、使用窗口规范,窗口规范支持以下格式:

  1. (ROWS | RANGE) BETWEEN (UNBOUNDED | [num]) PRECEDING AND ([num] PRECEDING | CURRENT ROW | (UNBOUNDED | [num]) FOLLOWING)
  2. (ROWS | RANGE) BETWEEN CURRENT ROW AND (CURRENT ROW | (UNBOUNDED | [num]) FOLLOWING)
  3. (ROWS | RANGE) BETWEEN [num] FOLLOWING AND (UNBOUNDED | [num]) FOLLOWING

当ORDER BY后面缺少窗口从句条件,窗口规范默认是 RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW.

当ORDER BY和窗口从句都缺失, 窗口规范默认是 ROW BETWEEN UNBOUNDED PRECEDING AND UNBOUNDED FOLLOWING.

OVER从句支持以下函数, 但是并不支持和窗口一起使用它们。 
Ranking函数: Rank, NTile, DenseRank, CumeDist, PercentRank. 
Lead 和 Lag 函数.

分析函数

ROW_NUMBER() 从1开始,按照顺序,生成分组内记录的序列,比如,按照pv降序排列,生成分组内每天的pv名次,ROW_NUMBER()的应用场景非常多,再比如,获取分组内排序第一的记录;获取一个session中的第一条refer等。 
RANK() 生成数据项在分组中的排名,排名相等会在名次中留下空位 
DENSE_RANK() 生成数据项在分组中的排名,排名相等会在名次中不会留下空位 
CUME_DIST 小于等于当前值的行数/分组内总行数。比如,统计小于等于当前薪水的人数,所占总人数的比例 
PERCENT_RANK 分组内当前行的RANK值-1/分组内总行数-1 
NTILE(n) 用于将分组数据按照顺序切分成n片,返回当前切片值,如果切片不均匀,默认增加第一个切片的分布。NTILE不支持ROWS BETWEEN,比如 NTILE(2) OVER(PARTITION BY cookieid ORDER BY createtime ROWS BETWEEN 3 PRECEDING AND CURRENT ROW)。

Hive2.1.0及以后支持Distinct

在聚合函数(SUM, COUNT and AVG)中,支持distinct,但是在ORDER BY 或者 窗口限制不支持。

COUNT(DISTINCT a) OVER (PARTITION BY c)

Hive 2.2.0中在使用ORDER BY和窗口限制时支持distinct

COUNT(DISTINCT a) OVER (PARTITION BY c ORDER BY d ROWS BETWEEN 1 PRECEDING AND 1 FOLLOWING)

Hive2.1.0及以后支持在OVER从句中支持聚合函数

  1. SELECT rank() OVER (ORDER BY sum(b))
  2. FROM T
  3. GROUP BY a;

测试数据集: 

  1. ## COUNTSUM、MIN、MAX、AVG
  2. select 
  3.     user_id,
  4.     user_type,
  5.     sales,
  6.     --默认为从起点到当前行
  7.     sum(sales) OVER(PARTITION BY user_type ORDER BY sales asc) AS sales_1,
  8.     --从起点到当前行,结果与sales_1不同。
  9.     sum(sales) OVER(PARTITION BY user_type ORDER BY sales asc ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW) AS sales_2,
  10.     --当前行+往前3
  11.     sum(sales) OVER(PARTITION BY user_type ORDER BY sales asc ROWS BETWEEN 3 PRECEDING AND CURRENT ROW) AS sales_3,
  12.     --当前行+往前3+往后1
  13.     sum(sales) OVER(PARTITION BY user_type ORDER BY sales asc ROWS BETWEEN 3 PRECEDING AND 1 FOLLOWING) AS sales_4,
  14.     --当前行+往后所有行  
  15.     sum(sales) OVER(PARTITION BY user_type ORDER BY sales asc ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING) AS sales_5,
  16.     --分组内所有行
  17.     SUM(sales) OVER(PARTITION BY user_type) AS sales_6                          
  18. from 
  19.     order_detail
  20. order by 
  21.     user_type,
  22.     sales,
  23.     user_id
  24. +----------+------------+--------+----------+----------+----------+----------+----------+----------+--+
  25. | user_id  | user_type  | sales  | sales_1  | sales_2  | sales_3  | sales_4  | sales_5  | sales_6  |
  26. +----------+------------+--------+----------+----------+----------+----------+----------+----------+--+
  27. | liiu     | new        | 1      | 2        | 2        | 2        | 4        | 22       | 23       |
  28. | qibaqiu  | new        | 1      | 2        | 1        | 1        | 2        | 23       | 23       |
  29. | zhangsa  | new        | 2      | 4        | 4        | 4        | 7        | 21       | 23       |
  30. | wanger   | new        | 3      | 7        | 7        | 7        | 12       | 19       | 23       |
  31. | lilisi   | new        | 5      | 17       | 17       | 15       | 21       | 11       | 23       |
  32. | qishili  | new        | 5      | 17       | 12       | 11       | 16       | 16       | 23       |
  33. | wutong   | new        | 6      | 23       | 23       | 19       | 19       | 6        | 23       |
  34. | lisi     | old        | 1      | 1        | 1        | 1        | 3        | 6        | 6        |
  35. | wangshi  | old        | 2      | 3        | 3        | 3        | 6        | 5        | 6        |
  36. | liwei    | old        | 3      | 6        | 6        | 6        | 6        | 3        | 6        |
  37. +----------+------------+--------+----------+----------+----------+----------+----------+----------+--+
  38. 注意:
  39. 结果和ORDER BY相关,默认为升序
  40. 如果不指定ROWS BETWEEN,默认为从起点到当前行;
  41. 如果不指定ORDER BY,则将分组内所有值累加;
  42. 关键是理解ROWS BETWEEN含义,也叫做WINDOW子句:
  43. PRECEDING:往前
  44. FOLLOWING:往后
  45. CURRENT ROW:当前行
  46. UNBOUNDED:无界限(起点或终点)
  47. UNBOUNDED PRECEDING:表示从前面的起点 
  48. UNBOUNDED FOLLOWING:表示到后面的终点
  49. 其他COUNT、AVG,MIN,MAX,和SUM用法一样。
  1. ## first_valuelast_value
  2. select 
  3.     user_id,
  4.     user_type,
  5.     ROW_NUMBER() OVER(PARTITION BY user_type ORDER BY sales) AS row_num,  
  6.     first_value(user_id) over (partition by user_type order by sales desc) as max_sales_user,
  7.     first_value(user_id) over (partition by user_type order by sales asc) as min_sales_user,
  8.     last_value(user_id) over (partition by user_type order by sales desc) as curr_last_min_user,
  9.     last_value(user_id) over (partition by user_type order by sales asc) as curr_last_max_user
  10. from 
  11.     order_detail;
  12. +----------+------------+----------+-----------------+-----------------+---------------------+---------------------+--+
  13. | user_id  | user_type  | row_num  | max_sales_user  | min_sales_user  | curr_last_min_user  | curr_last_max_user  |
  14. +----------+------------+----------+-----------------+-----------------+---------------------+---------------------+--+
  15. | wutong   | new        | 7        | wutong          | qibaqiu         | wutong              | wutong              |
  16. | lilisi   | new        | 6        | wutong          | qibaqiu         | qishili             | lilisi              |
  17. | qishili  | new        | 5        | wutong          | qibaqiu         | qishili             | lilisi              |
  18. | wanger   | new        | 4        | wutong          | qibaqiu         | wanger              | wanger              |
  19. | zhangsa  | new        | 3        | wutong          | qibaqiu         | zhangsa             | zhangsa             |
  20. | liiu     | new        | 2        | wutong          | qibaqiu         | qibaqiu             | liiu                |
  21. | qibaqiu  | new        | 1        | wutong          | qibaqiu         | qibaqiu             | liiu                |
  22. | liwei    | old        | 3        | liwei           | lisi            | liwei               | liwei               |
  23. | wangshi  | old        | 2        | liwei           | lisi            | wangshi             | wangshi             |
  24. | lisi     | old        | 1        | liwei           | lisi            | lisi                | lisi                |
  25. +----------+------------+----------+-----------------+-----------------+---------------------+---------------------+--+
  1. ## lead与lag
  2. select 
  3.     user_id,device_id,
  4.     lead(device_id) over (order by sales) as default_after_one_line,
  5.     lag(device_id) over (order by sales) as default_before_one_line,
  6.     lead(device_id,2) over (order by sales) as after_two_line,
  7.     lag(device_id,2,'abc') over (order by sales) as before_two_line
  8. from 
  9.     order_detail;
  10. +----------+-------------+-------------------------+--------------------------+-----------------+------------------+--+
  11. | user_id  |  device_id  | default_after_one_line  | default_before_one_line  | after_two_line  | before_two_line  |
  12. +----------+-------------+-------------------------+--------------------------+-----------------+------------------+--+
  13. | qibaqiu  | fds         | fdsfagwe                | NULL                     | 543gfd          | abc              |
  14. | liiu     | fdsfagwe    | 543gfd                  | fds                      | f332            | abc              |
  15. | lisi     | 543gfd      | f332                    | fdsfagwe                 | dfsadsa323      | fds              |
  16. | wangshi  | f332        | dfsadsa323              | 543gfd                   | hfd             | fdsfagwe         |
  17. | zhangsa  | dfsadsa323  | hfd                     | f332                     | 65ghf           | 543gfd           |
  18. | liwei    | hfd         | 65ghf                   | dfsadsa323               | fds             | f332             |
  19. | wanger   | 65ghf       | fds                     | hfd                      | dsfgg           | dfsadsa323       |
  20. | qishili  | fds         | dsfgg                   | 65ghf                    | 543gdfsd        | hfd              |
  21. | lilisi   | dsfgg       | 543gdfsd                | fds                      | NULL            | 65ghf            |
  22. | wutong   | 543gdfsd    | NULL                    | dsfgg                    | NULL            | fds              |
  23. +----------+-------------+-------------------------+--------------------------+-----------------+------------------+--+
  1. ## RANK、ROW_NUMBER、DENSE_RANK
  2. select 
  3.     user_id,user_type,sales,
  4.     RANK() over (partition by user_type order by sales desc) as r,
  5.     ROW_NUMBER() over (partition by user_type order by sales desc) as rn,
  6.     DENSE_RANK() over (partition by user_type order by sales desc) as dr
  7. from
  8.     order_detail;   
  9. +----------+------------+--------+----+-----+-----+--+
  10. | user_id  | user_type  | sales  | r  | rn  | dr  |
  11. +----------+------------+--------+----+-----+-----+--+
  12. | wutong   | new        | 6      | 1  | 1   | 1   |
  13. | qishili  | new        | 5      | 2  | 2   | 2   |
  14. | lilisi   | new        | 5      | 2  | 3   | 2   |
  15. | wanger   | new        | 3      | 4  | 4   | 3   |
  16. | zhangsa  | new        | 2      | 5  | 5   | 4   |
  17. | qibaqiu  | new        | 1      | 6  | 6   | 5   |
  18. | liiu     | new        | 1      | 6  | 7   | 5   |
  19. | liwei    | old        | 3      | 1  | 1   | 1   |
  20. | wangshi  | old        | 2      | 2  | 2   | 2   |
  21. | lisi     | old        | 1      | 3  | 3   | 3   |
  22. +----------+------------+--------+----+-----+-----+--+  
  1. ## NTILE
  2. select 
  3.     user_type,sales,
  4.     --分组内将数据分成2
  5.     NTILE(2) OVER(PARTITION BY user_type ORDER BY sales) AS nt2,
  6.     --分组内将数据分成3片    
  7.     NTILE(3) OVER(PARTITION BY user_type ORDER BY sales) AS nt3,
  8.     --分组内将数据分成4片    
  9.     NTILE(4) OVER(PARTITION BY user_type ORDER BY sales) AS nt4,
  10.     --将所有数据分成4
  11.     NTILE(4) OVER(ORDER BY sales) AS all_nt4
  12. from 
  13.     order_detail
  14. order by 
  15.     user_type,
  16.     sales
  17. +------------+--------+------+------+------+----------+--+
  18. | user_type  | sales  | nt2  | nt3  | nt4  | all_nt4  |
  19. +------------+--------+------+------+------+----------+--+
  20. | new        | 1      | 1    | 1    | 1    | 1        |
  21. | new        | 1      | 1    | 1    | 1    | 1        |
  22. | new        | 2      | 1    | 1    | 2    | 2        |
  23. | new        | 3      | 1    | 2    | 2    | 3        |
  24. | new        | 5      | 2    | 2    | 3    | 4        |
  25. | new        | 5      | 2    | 3    | 3    | 3        |
  26. | new        | 6      | 2    | 3    | 4    | 4        |
  27. | old        | 1      | 1    | 1    | 1    | 1        |
  28. | old        | 2      | 1    | 2    | 2    | 2        |
  29. | old        | 3      | 2    | 3    | 3    | 2        |
  30. +------------+--------+------+------+------+----------+--+
  31. 求取sale前20%的用户ID
  32. select
  33.     user_id
  34. from
  35. (
  36.     select 
  37.         user_id,
  38.         NTILE(5) OVER(ORDER BY sales desc) AS nt
  39.     from 
  40.         order_detail
  41. )A
  42. where nt=1;
  1. ## CUME_DIST、PERCENT_RANK 
  2. select 
  3. user_id,user_type,sales,
  4. --没有partition,所有数据均为1
  5. CUME_DIST() OVER(ORDER BY sales) AS cd1,
  6. --按照user_type进行分组
  7. CUME_DIST() OVER(PARTITION BY user_type ORDER BY sales) AS cd2 
  8. from 
  9. order_detail;   
  10. +----------+------------+--------+------+----------------------+--+
  11. | user_id  | user_type  | sales  | cd1  |         cd2          |
  12. +----------+------------+--------+------+----------------------+--+
  13. | liiu     | new        | 1      | 0.3  | 0.2857142857142857   |
  14. | qibaqiu  | new        | 1      | 0.3  | 0.2857142857142857   |
  15. | zhangsa  | new        | 2      | 0.5  | 0.42857142857142855  |
  16. | wanger   | new        | 3      | 0.7  | 0.5714285714285714   |
  17. | lilisi   | new        | 5      | 0.9  | 0.8571428571428571   |
  18. | qishili  | new        | 5      | 0.9  | 0.8571428571428571   |
  19. | wutong   | new        | 6      | 1.0  | 1.0                  |
  20. | lisi     | old        | 1      | 0.3  | 0.3333333333333333   |
  21. | wangshi  | old        | 2      | 0.5  | 0.6666666666666666   |
  22. | liwei    | old        | 3      | 0.7  | 1.0                  |
  23. +----------+------------+--------+------+----------------------+--+
  24. select 
  25. user_type,sales
  26. --分组内总行数      
  27. SUM(1) OVER(PARTITION BY user_type) AS s, 
  28. --RANK值  
  29. RANK() OVER(ORDER BY sales) AS r,    
  30. PERCENT_RANK() OVER(ORDER BY sales) AS pr,
  31. --分组内     
  32. PERCENT_RANK() OVER(PARTITION BY user_type ORDER BY sales) AS prg 
  33. from 
  34. order_detail;   
  35. +----+-----+---------------------+---------------------+--+
  36. | s  |  r  |         pr          |         prg         |
  37. +----+-----+---------------------+---------------------+--+
  38. | 7  | 1   | 0.0                 | 0.0                 |
  39. | 7  | 1   | 0.0                 | 0.0                 |
  40. | 7  | 4   | 0.3333333333333333  | 0.3333333333333333  |
  41. | 7  | 6   | 0.5555555555555556  | 0.5                 |
  42. | 7  | 8   | 0.7777777777777778  | 0.6666666666666666  |
  43. | 7  | 8   | 0.7777777777777778  | 0.6666666666666666  |
  44. | 7  | 10  | 1.0                 | 1.0                 |
  45. | 3  | 1   | 0.0                 | 0.0                 |
  46. | 3  | 4   | 0.3333333333333333  | 0.5                 |
  47. | 3  | 6   | 0.5555555555555556  | 1.0                 |
  48. +----+-----+---------------------+---------------------+--+

增强的聚合 Cube和Grouping 和Rollup

这几个分析函数通常用于OLAP中,不能累加,而且需要根据不同维度上钻和下钻的指标统计,比如,分小时、天、月的UV数。

GROUPING SETS 
在一个GROUP BY查询中,根据不同的维度组合进行聚合,等价于将不同维度的GROUP BY结果集进行UNION ALL, 
其中的GROUPING__ID,表示结果属于哪一个分组集合。

  1. select
  2.     user_type,
  3.     sales,
  4.     count(user_id) as pv,
  5.     GROUPING__ID 
  6. from 
  7.     order_detail
  8. group by 
  9.     user_type,sales
  10. GROUPING SETS(user_type,sales) 
  11. ORDER BY 
  12.     GROUPING__ID;
  13. +------------+--------+-----+---------------+--+
  14. | user_type  | sales  | pv  | grouping__id  |
  15. +------------+--------+-----+---------------+--+
  16. | old        | NULL   | 3   | 1             |
  17. | new        | NULL   | 7   | 1             |
  18. | NULL       | 6      | 1   | 2             |
  19. | NULL       | 5      | 2   | 2             |
  20. | NULL       | 3      | 2   | 2             |
  21. | NULL       | 2      | 2   | 2             |
  22. | NULL       | 1      | 3   | 2             |
  23. +------------+--------+-----+---------------+--+
  24. select
  25.     user_type,
  26.     sales,
  27.     count(user_id) as pv,
  28.     GROUPING__ID 
  29. from 
  30.     order_detail
  31. group by 
  32.     user_type,sales
  33. GROUPING SETS(user_type,sales,(user_type,sales)) 
  34. ORDER BY 
  35.     GROUPING__ID;
  36. +------------+--------+-----+---------------+--+
  37. | user_type  | sales  | pv  | grouping__id  |
  38. +------------+--------+-----+---------------+--+
  39. | old        | NULL   | 3   | 1             |
  40. | new        | NULL   | 7   | 1             |
  41. | NULL       | 1      | 3   | 2             |
  42. | NULL       | 6      | 1   | 2             |
  43. | NULL       | 5      | 2   | 2             |
  44. | NULL       | 3      | 2   | 2             |
  45. | NULL       | 2      | 2   | 2             |
  46. | old        | 3      | 1   | 3             |
  47. | old        | 2      | 1   | 3             |
  48. | old        | 1      | 1   | 3             |
  49. | new        | 6      | 1   | 3             |
  50. | new        | 5      | 2   | 3             |
  51. | new        | 3      | 1   | 3             |
  52. | new        | 1      | 2   | 3             |
  53. | new        | 2      | 1   | 3             |
  54. +------------+--------+-----+---------------+--+

CUBE 
根据GROUP BY的维度的所有组合进行聚合。

  1. select
  2.     user_type,
  3.     sales,
  4.     count(user_id) as pv,
  5.     GROUPING__ID 
  6. from 
  7.     order_detail
  8. group by 
  9.     user_type,sales
  10. WITH CUBE 
  11. ORDER BY 
  12.     GROUPING__ID;
  13. +------------+--------+-----+---------------+--+
  14. | user_type  | sales  | pv  | grouping__id  |
  15. +------------+--------+-----+---------------+--+
  16. | NULL       | NULL   | 10  | 0             |
  17. | new        | NULL   | 7   | 1             |
  18. | old        | NULL   | 3   | 1             |
  19. | NULL       | 6      | 1   | 2             |
  20. | NULL       | 5      | 2   | 2             |
  21. | NULL       | 3      | 2   | 2             |
  22. | NULL       | 2      | 2   | 2             |
  23. | NULL       | 1      | 3   | 2             |
  24. | old        | 3      | 1   | 3             |
  25. | old        | 2      | 1   | 3             |
  26. | old        | 1      | 1   | 3             |
  27. | new        | 6      | 1   | 3             |
  28. | new        | 5      | 2   | 3             |
  29. | new        | 3      | 1   | 3             |
  30. | new        | 2      | 1   | 3             |
  31. | new        | 1      | 2   | 3             |
  32. +------------+--------+-----+---------------+--+

ROLLUP 
是CUBE的子集,以最左侧的维度为主,从该维度进行层级聚合。

  1. select
  2.     user_type,
  3.     sales,
  4.     count(user_id) as pv,
  5.     GROUPING__ID 
  6. from 
  7.     order_detail
  8. group by 
  9.     user_type,sales
  10. WITH ROLLUP 
  11. ORDER BY 
  12.     GROUPING__ID;
  13. +------------+--------+-----+---------------+--+
  14. | user_type  | sales  | pv  | grouping__id  |
  15. +------------+--------+-----+---------------+--+
  16. | NULL       | NULL   | 10  | 0             |
  17. | old        | NULL   | 3   | 1             |
  18. | new        | NULL   | 7   | 1             |
  19. | old        | 3      | 1   | 3             |
  20. | old        | 2      | 1   | 3             |
  21. | old        | 1      | 1   | 3             |
  22. | new        | 6      | 1   | 3             |
  23. | new        | 5      | 2   | 3             |
  24. | new        | 3      | 1   | 3             |
  25. | new        | 2      | 1   | 3             |
  26. | new        | 1      | 2   | 3             |
  27. +------------+--------+-----+---------------+--+

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/IT小白/article/detail/656575
推荐阅读
相关标签
  

闽ICP备14008679号