1. 程式人生 > >Hive SQL綜合案例

Hive SQL綜合案例

嚴格 存在 排序規則 連接 是我 語文課 sub 規則 年齡

一 Hive SQL練習之影評案例

案例說明

現有如此三份數據:
1、users.dat 數據格式為: 2::M::56::16::70072,

共有6040條數據
對應字段為:UserID BigInt, Gender String, Age Int, Occupation String, Zipcode String
對應字段中文解釋:用戶id,性別,年齡,職業,郵政編碼

2、movies.dat 數據格式為: 2::Jumanji (1995)::Adventure|Children‘s|Fantasy,

共有3883條數據
對應字段為:MovieID BigInt, Title String, Genres String

對應字段中文解釋:電影ID,電影名字,電影類型

3、ratings.dat 數據格式為: 1::1193::5::978300760,

共有1000209條數據
對應字段為:UserID BigInt, MovieID BigInt, Rating Double, Timestamped String
對應字段中文解釋:用戶ID,電影ID,評分,評分時間戳

題目要求

  數據要求:
    (1)寫shell腳本清洗數據。(hive不支持解析多字節的分隔符,也就是說hive只能解析‘:‘, 不支持解析‘::‘,所以用普通方式建表來使用是行不通的,要求對數據做一次簡單清洗)
    (2)使用Hive能解析的方式進行

  Hive要求:
    (1)正確建表,導入數據(三張表,三份數據),並驗證是否正確

    (2)求被評分次數最多的10部電影,並給出評分次數(電影名,評分次數)

    (3)分別求男性,女性當中評分最高的10部電影(性別,電影名,影評分)

    (4)求movieid = 2116這部電影各年齡段(因為年齡就只有7個,就按這個7個分就好了)的平均影評(年齡段,影評分)

    (5)求最喜歡看電影(影評次數最多)的那位女性評最高分的10部電影的平均影評分(觀影者,電影名,影評分)

    (6)求好片(評分>=4.0)最多的那個年份的最好看的10部電影

    (7)求1997年上映的電影中,評分最高的10部Comedy類電影

    (8)該影評庫中各種類型電影中評價最高的5部電影(類型,電影名,平均影評分)

    (9)各年評分最高的電影類型(年份,類型,影評分)

    (10)每個地區最高評分的電影名,把結果存入HDFS(地區,電影名,影評分)

數據下載

https://files.cnblogs.com/files/qingyunzong/hive%E5%BD%B1%E8%AF%84%E6%A1%88%E4%BE%8B.zip

解析

之前已經使用MapReduce程序將3張表格進行合並,所以只需要將合並之後的表格導入對應的表中進行查詢即可。

1、正確建表,導入數據(三張表,三份數據),並驗證是否正確

(1)分析需求

需要創建一個數據庫movie,在movie數據庫中創建3張表,t_user,t_movie,t_rating

t_user:userid bigint,sex string,age int,occupation string,zipcode string
t_movie:movieid bigint,moviename string,movietype string
t_rating:userid bigint,movieid bigint,rate double,times string

原始數據是以::進行切分的,所以需要使用能解析多字節分隔符的Serde即可

使用RegexSerde

需要兩個參數:
input.regex = "(.*)::(.*)::(.*)"
output.format.string = "%1$s %2$s %3$s"

(2)創建數據庫

drop database if exists movie;
create database if not exists movie;
use movie;

(3)創建t_user表

create table t_user(
userid bigint,
sex string,
age int,
occupation string,
zipcode string) 
row format serde ‘org.apache.hadoop.hive.serde2.RegexSerDe‘ 
with serdeproperties(‘input.regex‘=‘(.*)::(.*)::(.*)::(.*)::(.*)‘,‘output.format.string‘=‘%1$s %2$s %3$s %4$s %5$s‘)
stored as textfile;

(4)創建t_movie表

use movie;
create table t_movie(
movieid bigint,
moviename string,
movietype string) 
row format serde ‘org.apache.hadoop.hive.serde2.RegexSerDe‘ 
with serdeproperties(‘input.regex‘=‘(.*)::(.*)::(.*)‘,‘output.format.string‘=‘%1$s %2$s %3$s‘)
stored as textfile;

(5)創建t_rating表

use movie;
create table t_rating(
userid bigint,
movieid bigint,
rate double,
times string) 
row format serde ‘org.apache.hadoop.hive.serde2.RegexSerDe‘ 
with serdeproperties(‘input.regex‘=‘(.*)::(.*)::(.*)::(.*)‘,‘output.format.string‘=‘%1$s %2$s %3$s %4$s‘)
stored as textfile;

(6)導入數據

0: jdbc:hive2://hadoop3:10000> load data local inpath "/home/hadoop/movie/users.dat" into table t_user;
No rows affected (0.928 seconds)
0: jdbc:hive2://hadoop3:10000> load data local inpath "/home/hadoop/movie/movies.dat" into table t_movie;
No rows affected (0.538 seconds)
0: jdbc:hive2://hadoop3:10000> load data local inpath "/home/hadoop/movie/ratings.dat" into table t_rating;
No rows affected (0.963 seconds)
0: jdbc:hive2://hadoop3:10000> 

(7)驗證

select t.* from t_user t;

技術分享圖片

select t.* from t_movie t;

技術分享圖片

select t.* from t_rating t;

技術分享圖片

2、求被評分次數最多的10部電影,並給出評分次數(電影名,評分次數)

(1)思路分析:

  1、需求字段:電影名 t_movie.moviename

         評分次數 t_rating.rate count()

  2、核心SQL:按照電影名進行分組統計,求出每部電影的評分次數並按照評分次數降序排序

(2)完整SQL:

create table answer2 as 
select a.moviename as moviename,count(a.moviename) as total 
from t_movie a join t_rating b on a.movieid=b.movieid 
group by a.moviename 
order by total desc 
limit 10;
select * from answer2;

技術分享圖片

3、分別求男性,女性當中評分最高的10部電影(性別,電影名,影評分)

(1)分析思路:

  1、需求字段:性別  t_user.sex

         電影名 t_movie.moviename

         影評分 t_rating.rate

  2、核心SQL:三表聯合查詢,按照性別過濾條件,電影名作為分組條件,影評分作為排序條件進行查詢

(2)完整SQL:

女性當中評分最高的10部電影(性別,電影名,影評分)評論次數大於等於50次

create table answer3_F as 
select "F" as sex, c.moviename as name, avg(a.rate) as avgrate, count(c.moviename) as total  
from t_rating a 
join t_user b on a.userid=b.userid 
join t_movie c on a.movieid=c.movieid 
where b.sex="F" 
group by c.moviename 
having total >= 50
order by avgrate desc 
limit 10;
select * from answer3_F;

技術分享圖片

男性當中評分最高的10部電影(性別,電影名,影評分)評論次數大於等於50次

create table answer3_M as 
select "M" as sex, c.moviename as name, avg(a.rate) as avgrate, count(c.moviename) as total  
from t_rating a 
join t_user b on a.userid=b.userid 
join t_movie c on a.movieid=c.movieid 
where b.sex="M" 
group by c.moviename 
having total >= 50
order by avgrate desc 
limit 10;
select * from answer3_M;

技術分享圖片

4、求movieid = 2116這部電影各年齡段(因為年齡就只有7個,就按這個7個分就好了)的平均影評(年齡段,影評分)

(1)分析思路:

  1、需求字段:年齡段  t_user.age

         影評分 t_rating.rate

  2、核心SQL:t_user和t_rating表進行聯合查詢,用movieid=2116作為過濾條件,用年齡段作為分組條件

(2)完整SQL:

create table answer4 as 
select a.age as age, avg(b.rate) as avgrate 
from t_user a join t_rating b on a.userid=b.userid 
where b.movieid=2116 
group by a.age;
select * from answer4;

技術分享圖片

5、求最喜歡看電影(影評次數最多)的那位女性評最高分的10部電影的平均影評分(觀影者,電影名,影評分)

(1)分析思路:

  1、需求字段:觀影者 t_rating.userid

         電影名 t_movie.moviename

         影評分 t_rating.rate

  2、核心SQL:

    A.  需要先求出最喜歡看電影的那位女性

          需要查詢的字段:性別:t_user.sex

                  觀影次數:count(t_rating.userid)

    B.  根據A中求出的女性userid作為where過濾條件,以看過的電影的影評分rate作為排序條件進行排序,求出評分最高的10部電影

          需要查詢的字段:電影的ID:t_rating.movieid

    C.  求出B中10部電影的平均影評分

          需要查詢的字段:電影的ID:answer5_B.movieid

                  影評分:t_rating.rate

(2)完整SQL:

A.  需要先求出最喜歡看電影的那位女性

select a.userid, count(a.userid) as total 
from t_rating a join t_user b on a.userid = b.userid 
where b.sex="F" 
group by a.userid 
order by total desc 
limit 1;

技術分享圖片

B.  根據A中求出的女性userid作為where過濾條件,以看過的電影的影評分rate作為排序條件進行排序,求出評分最高的10部電影

create table answer5_B as 
select a.movieid as movieid, a.rate as rate  
from t_rating a 
where a.userid=1150 
order by rate desc 
limit 10;
select * from answer5_B;

技術分享圖片

C.  求出B中10部電影的平均影評分

create table answer5_C as 
select b.movieid as movieid, c.moviename as moviename, avg(b.rate) as avgrate 
from answer5_B a 
join t_rating b on a.movieid=b.movieid 
join t_movie c on b.movieid=c.movieid 
group by b.movieid,c.moviename;
select * from answer5_C;

技術分享圖片

6、求好片(評分>=4.0)最多的那個年份的最好看的10部電影

(1)分析思路:

  1、需求字段:電影id t_rating.movieid

         電影名 t_movie.moviename(包含年份)

         影評分 t_rating.rate

         上映年份 xxx.years

  2、核心SQL:

    A.  需要將t_rating和t_movie表進行聯合查詢,將電影名當中的上映年份截取出來,保存到臨時表answer6_A中

          需要查詢的字段:電影id t_rating.movieid

                  電影名 t_movie.moviename(包含年份)

                  影評分 t_rating.rate

    B.  從answer6_A按照年份進行分組條件,按照評分>=4.0作為where過濾條件,按照count(years)作為排序條件進行查詢

          需要查詢的字段:電影的ID:answer6_A.years

    C.  從answer6_A按照years=1998作為where過濾條件,按照評分作為排序條件進行查詢

          需要查詢的字段:電影的ID:answer6_A.moviename

                  影評分:answer6_A.avgrate

(2)完整SQL:

A.  需要將t_rating和t_movie表進行聯合查詢,將電影名當中的上映年份截取出來

create table answer6_A as
select  a.movieid as movieid, a.moviename as moviename, substr(a.moviename,-5,4) as years, avg(b.rate) as avgrate
from t_movie a join t_rating b on a.movieid=b.movieid 
group by a.movieid, a.moviename;
select * from answer6_A;

技術分享圖片

B.  從answer6_A按照年份進行分組條件,按照評分>=4.0作為where過濾條件,按照count(years)作為排序條件進行查詢

select years, count(years) as total 
from answer6_A a 
where avgrate >= 4.0 
group by years 
order by total desc 
limit 1;

技術分享圖片

C.  從answer6_A按照years=1998作為where過濾條件,按照評分作為排序條件進行查詢

create table answer6_C as
select a.moviename as name, a.avgrate as rate 
from answer6_A a 
where a.years=1998 
order by rate desc 
limit 10;
select * from answer6_C;

技術分享圖片

7、求1997年上映的電影中,評分最高的10部Comedy類電影

(1)分析思路:

  1、需求字段:電影id t_rating.movieid

         電影名 t_movie.moviename(包含年份)

         影評分 t_rating.rate

         上映年份 xxx.years(最終查詢結果可不顯示)

         電影類型 xxx.type(最終查詢結果可不顯示)

  2、核心SQL:

    A.  需要電影類型,所有可以將第六步中求出answer6_A表和t_movie表進行聯合查詢

          需要查詢的字段:電影id answer6_A.movieid

                  電影名 answer6_A.moviename

                  影評分 answer6_A.rate

                  電影類型 t_movie.movietype 

                  上映年份 answer6_A.years

    B.  從answer7_A按照電影類型中是否包含Comedy和按上映年份作為where過濾條件,按照評分作為排序條件進行查詢,將結果保存到answer7_B中

          需要查詢的字段:電影的ID:answer7_A.id

                  電影的名稱:answer7_A.name

                  電影的評分:answer7_A.rate

(2)完整SQL:

A.  需要電影類型,所有可以將第六步中求出answer6_A表和t_movie表進行聯合查詢

create table answer7_A as 
select b.movieid as id, b.moviename as name, b.years as years, b.avgrate as rate, a.movietype as type 
from t_movie a join answer6_A b on a.movieid=b.movieid;
select t.* from answer7_A t;

技術分享圖片

B.  從answer7_A按照電影類型中是否包含Comedy和按照評分>=4.0作為where過濾條件,按照評分作為排序條件進行查詢,將結果保存到answer7_B中

create table answer7_B as 
select t.id as id, t.name as name, t.rate as rate 
from answer7_A t 
where t.years=1997 and instr(lcase(t.type),‘comedy‘) >0 
order by rate desc
limit 10;
select * from answer7_B;

技術分享圖片

8、該影評庫中各種類型電影中評價最高的5部電影(類型,電影名,平均影評分)

(1)分析思路:

  1、需求字段:電影id movieid

         電影名 moviename

         影評分 rate(排序條件)

         電影類型 type(分組條件)

  2、核心SQL:

    A.  需要電影類型,所有需要將answer7_A中的type字段進行裂變,將結果保存到answer8_A中

          需要查詢的字段:電影id answer7_A.id

                  電影名 answer7_A.name(包含年份)

                  上映年份 answer7_A.years

                  影評分 answer7_A.rate

                  電影類型 answer7_A.movietype 

    B.  求TopN,按照type分組,需要添加一列來記錄每組的順序,將結果保存到answer8_B中

row_number() :用來生成 num字段的值

distribute by movietype :按照type進行分組

sort by avgrate desc :每組數據按照rate排降序

num:新列, 值就是每一條記錄在每一組中按照排序規則計算出來的排序值

    C.  從answer8_B中取出num列序號<=5的

(2)完整SQL:

A.  需要電影類型,所有需要將answer7_A中的type字段進行裂變,將結果保存到answer8_A中

create table answer8_A as 
select a.id as id, a.name as name, a.years as years, a.rate as rate, tv.type as type 
from answer7_A a 
lateral view explode(split(a.type,"\\|")) tv as type;
select * from answer8_A; 

技術分享圖片

B.  求TopN,按照type分組,需要添加一列來記錄每組的順序,將結果保存到answer8_B中

create table answer8_B as 
select id,name,years,rate,type,row_number() over(distribute by type sort by rate desc ) as num
from answer8_A;
select * from answer8_B;

技術分享圖片

C.  從answer8_B中取出num列序號<=5的

select a.* from answer8_B a where a.num <= 5;

技術分享圖片

9、各年評分最高的電影類型(年份,類型,影評分)

(1)分析思路:

  1、需求字段:電影id movieid

         電影名 moviename

         影評分 rate(排序條件)

         電影類型 type(分組條件)

         上映年份 years(分組條件)

  2、核心SQL:

    A.  需要按照電影類型和上映年份進行分組,按照影評分進行排序,將結果保存到answer9_A中

          需要查詢的字段:

                  上映年份 answer7_A.years

                  影評分 answer7_A.rate

                  電影類型 answer7_A.movietype 

    B.  求TopN,按照years分組,需要添加一列來記錄每組的順序,將結果保存到answer9_B中

    C.  按照num=1作為where過濾條件取出結果數據

(2)完整SQL:

A.  需要按照電影類型和上映年份進行分組,按照影評分進行排序,將結果保存到answer9_A中

create table answer9_A as 
select a.years as years, a.type as type, avg(a.rate) as rate 
from answer8_A a 
group by a.years,a.type 
order by rate desc;
select * from answer9_A;

技術分享圖片

B.  求TopN,按照years分組,需要添加一列來記錄每組的順序,將結果保存到answer9_B中

create table answer9_B as 
select years,type,rate,row_number() over (distribute by years sort by rate) as num
from answer9_A;
select * from answer9_B;

技術分享圖片

C.  按照num=1作為where過濾條件取出結果數據

select * from answer9_B where num=1;

技術分享圖片

10、每個地區最高評分的電影名,把結果存入HDFS(地區,電影名,影評分)

(1)分析思路:

  1、需求字段:電影id t_movie.movieid

         電影名 t_movie.moviename

         影評分 t_rating.rate(排序條件)

         地區 t_user.zipcode(分組條件)

  2、核心SQL:

    A.  需要把三張表進行聯合查詢,取出電影id、電影名稱、影評分、地區,將結果保存到answer10_A表中

          需要查詢的字段:電影id t_movie.movieid

                   電影名 t_movie.moviename

                   影評分 t_rating.rate(排序條件)

                   地區 t_user.zipcode(分組條件)

    B.  求TopN,按照地區分組,按照平均排序,添加一列num用來記錄地區排名,將結果保存到answer10_B表中

    C.  按照num=1作為where過濾條件取出結果數據

(2)完整SQL:

A.  需要把三張表進行聯合查詢,取出電影id、電影名稱、影評分、地區,將結果保存到answer10_A表中

create table answer10_A as
select c.movieid, c.moviename, avg(b.rate) as avgrate, a.zipcode
from t_user a 
join t_rating b on a.userid=b.userid 
join t_movie c on b.movieid=c.movieid 
group by a.zipcode,c.movieid, c.moviename;
select t.* from answer10_A t;

技術分享圖片

B.  求TopN,按照地區分組,按照平均排序,添加一列num用來記錄地區排名,將結果保存到answer10_B表中

create table answer10_B as
select movieid,moviename,avgrate,zipcode, row_number() over (distribute by zipcode sort by avgrate) as num 
from answer10_A; 
select t.* from answer10_B t;

技術分享圖片

C.  按照num=1作為where過濾條件取出結果數據並保存到HDFS上

insert overwrite directory "/movie/answer10/" select t.* from answer10_B t where t.num=1;

技術分享圖片

二、求單月訪問次數和總訪問次數

1、數據說明

數據字段說明

用戶名,月份,訪問次數

數據格式

A,2015-01,5
A,2015-01,15
B,2015-01,5
A,2015-01,8
B,2015-01,25
A,2015-01,5
A,2015-02,4
A,2015-02,6
B,2015-02,10
B,2015-02,5
A,2015-03,16
A,2015-03,22
B,2015-03,23
B,2015-03,10
B,2015-03,1

2、數據準備

(1)創建表

use myhive;
create external table if not exists t_access(
uname string comment ‘用戶名‘,
umonth string comment ‘月份‘,
ucount int comment ‘訪問次數‘
) comment ‘用戶訪問表‘ 
row format delimited fields terminated by "," 
location "/hive/t_access"; 

(2)導入數據

load data local inpath "/home/hadoop/access.txt" into table t_access;

(3)驗證數據

select * from t_access;

技術分享圖片

3、結果需求

現要求出:
每個用戶截止到每月為止的最大單月訪問次數和累計到該月的總訪問次數,結果數據格式如下

技術分享圖片

4、需求分析

此結果需要根據用戶+月份進行分組

(1)先求出當月訪問次數

--求當月訪問次數
create table tmp_access(
name string,
mon string,
num int
); 

insert into table tmp_access 
select uname,umonth,sum(ucount)
 from t_access t group by t.uname,t.umonth;

select * from tmp_access;

技術分享圖片

(2)tmp_access進行自連接視圖

create view tmp_view as 
select a.name anme,a.mon amon,a.num anum,b.name bname,b.mon bmon,b.num bnum from tmp_access a join tmp_access b 
on a.name=b.name;

select * from tmp_view;

技術分享圖片

(3)進行比較統計

select anme,amon,anum,max(bnum) as max_access,sum(bnum) as sum_access 
from tmp_view 
where amon>=bmon 
group by anme,amon,anum;

技術分享圖片

三、學生課程成績

1、數據準備

use myhive;
CREATE TABLE `course` (
  `id` int,
  `sid` int ,
  `course` string,
  `score` int 
) ;
// 插入數據
// 字段解釋:id, 學號, 課程, 成績
INSERT INTO `course` VALUES (1, 1, ‘yuwen‘, 43);
INSERT INTO `course` VALUES (2, 1, ‘shuxue‘, 55);
INSERT INTO `course` VALUES (3, 2, ‘yuwen‘, 77);
INSERT INTO `course` VALUES (4, 2, ‘shuxue‘, 88);
INSERT INTO `course` VALUES (5, 3, ‘yuwen‘, 98);
INSERT INTO `course` VALUES (6, 3, ‘shuxue‘, 65);

技術分享圖片

2、需求

求:所有數學課程成績 大於 語文課程成績的學生的學號

1、使用case...when...將不同的課程名稱轉換成不同的列

create view tmp_course_view as
select sid, case course when "shuxue" then score else 0 end  as shuxue,  
case course when "yuwen" then score else 0 end  as yuwen from course;  

select * from tmp_course_view;

技術分享圖片

2、以sid分組合並取各成績最大值

create view tmp_course_view1 as
select aa.sid, max(aa.shuxue) as shuxue, max(aa.yuwen) as yuwen from tmp_course_view aa group by sid;  

select * from tmp_course_view1;

技術分享圖片

3、比較結果

select * from tmp_course_view1 where shuxue > yuwen;

技術分享圖片

四、求每一年最大氣溫的那一天+溫度

1、說明

數據格式

2010012325

具體數據

技術分享圖片 View Code

數據解釋

2010012325表示在2010年01月23日的氣溫為25度

2、 需求

比如:2010012325表示在2010年01月23日的氣溫為25度。現在要求使用hive,計算每一年出現過的最大氣溫的日期+溫度。
要計算出每一年的最大氣溫。我用
select substr(data,1,4),max(substr(data,9,2)) from table2 group by substr(data,1,4);
出來的是 年份 + 溫度 這兩列數據例如 2015 99

但是如果我是想select 的是:具體每一年最大氣溫的那一天 + 溫度 。例如 20150109 99
請問該怎麽執行hive語句。。
group by 只需要substr(data,1,4),
但是select substr(data,1,8),又不在group by 的範圍內。
是我陷入了思維死角。一直想不出所以然。。求大神指點一下。
在select 如果所需要的。不在group by的條件裏。這種情況如何去分析?

3、解析

(1)創建一個臨時表tmp_weather,將數據切分

create table tmp_weather as 
select substr(data,1,4) years,substr(data,5,2) months,substr(data,7,2) days,substr(data,9,2) temp from weather;
select * from tmp_weather;

技術分享圖片

(2)創建一個臨時表tmp_year_weather

create table tmp_year_weather as 
select substr(data,1,4) years,max(substr(data,9,2)) max_temp from weather group by substr(data,1,4);
select * from tmp_year_weather;

技術分享圖片

(3)將2個臨時表進行連接查詢

select * from tmp_year_weather a join tmp_weather b on a.years=b.years and a.max_temp=b.temp;

技術分享圖片

五、求學生選課情況

1、數據說明

(1)數據格式

id course 
1,a 
1,b 
1,c 
1,e 
2,a 
2,c 
2,d 
2,f 
3,a 
3,b 
3,c 
3,e

(2)字段含義

表示有id為1,2,3的學生選修了課程a,b,c,d,e,f中其中幾門。

2、數據準備

(1)建表t_course

create table t_course(id int,course string)
row format delimited fields terminated by ",";

技術分享圖片

(2)導入數據

load data local inpath "/home/hadoop/course/course.txt" into table t_course;

技術分享圖片

3、需求

編寫Hive的HQL語句來實現以下結果:表中的1表示選修,表中的0表示未選修

id    a    b    c    d    e    f
1     1    1    1    0    1    0
2     1    0    1    1    0    1
3     1    1    1    0    1    0

4、解析

第一步:

select collect_set(course) as courses from id_course;

第二步:

set hive.strict.checks.cartesian.product=false;

create table id_courses as select t1.id as id,t1.course as id_courses,t2.course courses 
from 
( select id as id,collect_set(course) as course from id_course group by id ) t1 
join 
(select collect_set(course) as course from id_course) t2;

啟用嚴格模式:hive.mapred.mode = strict // Deprecated
hive.strict.checks.large.query = true
該設置會禁用:1. 不指定分頁的orderby
       2. 對分區表不指定分區進行查詢
       3. 和數據量無關,只是一個查詢模式

hive.strict.checks.type.safety = true
嚴格類型安全,該屬性不允許以下操作:1. bigint和string之間的比較
                  2. bigint和double之間的比較

hive.strict.checks.cartesian.product = true
該屬性不允許笛卡爾積操作

第三步:得出最終結果:
思路:
拿出course字段中的每一個元素在id_courses中進行判斷,看是否存在。

select id,
case when array_contains(id_courses, courses[0]) then 1 else 0 end as a,
case when array_contains(id_courses, courses[1]) then 1 else 0 end as b,
case when array_contains(id_courses, courses[2]) then 1 else 0 end as c,
case when array_contains(id_courses, courses[3]) then 1 else 0 end as d,
case when array_contains(id_courses, courses[4]) then 1 else 0 end as e,
case when array_contains(id_courses, courses[5]) then 1 else 0 end as f 
from id_courses;

六、求月銷售額和總銷售額

1、數據說明

(1)數據格式

a,01,150
a,01,200
b,01,1000
b,01,800
c,01,250
c,01,220
b,01,6000
a,02,2000
a,02,3000
b,02,1000
b,02,1500
c,02,350
c,02,280
a,03,350
a,03,250

(2)字段含義

店鋪,月份,金額

2、數據準備

(1)創建數據庫表t_store

use class;
create table t_store(
name string,
months int,
money int
) 
row format delimited fields terminated by ",";

(2)導入數據

load data local inpath "/home/hadoop/store.txt" into table t_store;

3、需求

編寫Hive的HQL語句求出每個店鋪的當月銷售額和累計到當月的總銷售額

4、解析

(1)按照商店名稱和月份進行分組統計

create table tmp_store1 as 
select name,months,sum(money) as money from t_store group by name,months;

select * from tmp_store1;

技術分享圖片

(2)對tmp_store1 表裏面的數據進行自連接

create table tmp_store2 as 
select a.name aname,a.months amonths,a.money amoney,b.name bname,b.months bmonths,b.money bmoney from tmp_store1 a 
join tmp_store1 b on a.name=b.name order by aname,amonths;

select * from tmp_store2;

技術分享圖片

(3)比較統計

select aname,amonths,amoney,sum(bmoney) as total from tmp_store2 where amonths >= bmonths group by aname,amonths,amoney;

技術分享圖片

Hive SQL綜合案例