转载出处:https://blog.csdn.net/u011726005/article/details/94592866
目录
1. 排名分类
1.1 区别RANK,DENSE_RANK和ROW_NUMBER
1.2 分组排名
2. 准备数据
3. 不分组排名
3.1 连续排名ROW_NUMBER
3.2 并列跳跃排名RANK
3.3 并列连续排名DENSE_RANK
4. 分组排名
4.1 分组连续排名ROW_NUMBER
4.2 分组并列跳跃排名RANK
4.3 分组并列连续排名DENSE_RANK
窗口函数不同于常规的聚合函数, 使用的窗口函数不会导致行被分组到一个单一的输出行;行保留其独立的身份。 在后台,窗口函数能够访问的不止查询结果的当前行。
在MYSQL的最新版本MYSQL8已经支持了排名函数RANK,DENSE_RANK和ROW_NUMBER。但是在就得版本中还不支持这些函数,只能自己实现。实现方法主要用到了条件判断语句(CASE WHEN或IF)和添加临时变量。
区别如图:
将数据分组后排名,区别如图:
创建一张分数表,里面有字段:分数score,课程号course_id和学生号student_id。
执行如下SQL语句,进行导入数据。
create table score(
student_id varchar(10),
course_id varchar(10),
score decimal(18,1)
);
insert into score values('01' , '01' , 80);
insert into score values('01' , '02' , 90);
insert into score values('01' , '03' , 99);
insert into score values('02' , '01' , 70);
insert into score values('02' , '02' , 60);
insert into score values('02' , '03' , 80);
insert into score values('03' , '01' , 80);
insert into score values('03' , '02' , 80);
insert into score values('03' , '03' , 80);
insert into score values('04' , '01' , 50);
insert into score values('04' , '02' , 30);
insert into score values('04' , '03' , 20);
insert into score values('05' , '01' , 76);
insert into score values('05' , '02' , 87);
insert into score values('06' , '01' , 31);
insert into score values('06' , '03' , 34);
insert into score values('07' , '02' , 89);
insert into score values('07' , '03' , 98);
insert into score values('08' , '02' , 89);
insert into score values('09' , '02' , 89);
查看数据:
SELECT
score,
ROW_NUMBER() OVER (ORDER BY score DESC) as rank
FROM
score;
SELECT
s.score,
(@cur_rank := @cur_rank + 1) ranking
FROM
score s,
(SELECT @cur_rank := 0) r
ORDER BY
score DESC;
结果如图:
SELECT
course_id,
score,
RANK() OVER(ORDER BY score DESC)
FROM
score;
# rank_counter:排名自增计数,一直往上加
# cur_rank:表示当前排名
# pre_score:表示前一个分数
# 每前进一条记录,rank_counter就加一
# 如果前一个分数等于当前分数,那就保持当前排名,如果不等于,直接跳到rank_counter的位置
# 要继续前进时,pre_score置为当前分数
SELECT
s.score,
@rank_counter := @rank_counter + 1,
IF(@pre_score = s.score, @cur_rank, @cur_rank := @rank_counter) ranking,
@pre_score := s.score
FROM
score s,
(SELECT @cur_rank :=0, @pre_score := NULL, @rank_counter := 0) r
ORDER BY
s.score DESC;
SELECT
s.score,
@rank_counter := @rank_counter + 1,
(
CASE
WHEN @pre_score = s.score THEN @cur_rank
WHEN @pre_score := s.score THEN @cur_rank := @rank_counter
END
) ranking
FROM
score s,
(SELECT @cur_rank :=0, @pre_score := NULL, @rank_counter := 0) r # 初始化参数
ORDER BY
s.score DESC;
结果如图:
SELECT
course_id,
score,
DENSE_RANK() OVER(ORDER BY score DESC)
FROM
score;
# cur_rank:表示当前排名
# pre_score:表示前一个分数
# 如果前一个分数等于当前分数,那就保持当前排名,如果不等于,+1
# 要继续前进时,pre_score置为当前分数
SELECT
s.score,
IF(@pre_score = s.score, @cur_rank, @cur_rank := @cur_rank + 1) ranking,
@pre_score := s.score
FROM
score s,
(SELECT @cur_rank :=0, @pre_score = NULL) r
ORDER BY
s.score DESC;
SELECT
s.score,
(
CASE
WHEN @pre_score = s.score THEN @cur_rank
WHEN @pre_score := s.score THEN @cur_rank := @cur_rank + 1
END
) ranking
FROM
score s,
(SELECT @cur_rank :=0, @pre_score = NULL) r
ORDER BY
s.score DESC;
结果如图:
PARTITION BY course_id ORDER BY score DESC) 其实就像普通order by course_id,score
SELECT
course_id,
score,
ROW_NUMBER() OVER (PARTITION BY course_id ORDER BY score DESC) ranking
FROM
score;
# cur_rank:表示当前排名
# pre_course_id:表示前一个课程ID
# 如果前一个课程ID等于当前课程ID(说明还是一门课的),+1,否则就将cur_rank置为1
# 要继续前进时,pre_score置为当前分数
使用 变量和IF语句 实现:
SELECT
s.course_id,
s.score,
IF(@pre_course_id = s.course_id, @cur_rank := @cur_rank + 1, @cur_rank := 1) ranking,
@pre_course_id := s.course_id
FROM
score s,
(SELECT @cur_rank := 0, @pre_course_id := NULL) r
ORDER BY
course_id,
score DESC;
结果如图:
SELECT
course_id,
score,
RANK() OVER(PARTITION BY course_id ORDER BY score DESC)
FROM
score;
# rank_counter:排名自增计数,一直往上加(不过此时要看前后记录是否是同一课程)
# cur_rank:表示当前排名# pre_course_id:表示前一个课程号
# pre_score:表示前一个分数
# 每前进一条记录,如果前后是同一课程,rank_counter就加一,否则就重置为1 这是temp1
# 如果前后是同一课程,而且前后分数相同,那cur_rank不变 ,如果前后分数不同,cur_rank直接跳到rank_counter的位置,如果前后不是同一课程,cur_rank置为1
# 要继续前进时,pre_score置为当前分数,pre_score_id置为当前课程id
SELECT
s.course_id,
s.score,
IF(@pre_course_id = s.course_id,
@rank_counter := @rank_counter + 1,
@rank_counter := 1) temp1,
IF(@pre_course_id = s.course_id,
IF(@pre_score = s.score, @cur_rank, @cur_rank := @rank_counter),
@cur_rank := 1) ranking,
@pre_score := s.score temp2,
@pre_course_id := s.course_id temp3
FROM
score s,
(
SELECT
@cur_rank := 0,
@pre_course_id := NULL,
@pre_score := NULL,
@rank_counter := 1
) r
ORDER BY
s.course_id,
s.score DESC;
结果如图:
SELECT
course_id,
score,
DENSE_RANK() OVER(PARTITION BY course_id ORDER BY score DESC)
FROM
score;
SELECT
s.course_id,
s.score,
IF(@pre_course_id = s.course_id,
IF(@pre_score = s.score, @cur_rank, @cur_rank := @cur_rank + 1),
@cur_rank := 1) ranking,
@pre_score := s.score,
@pre_course_id := s.course_id
FROM
score s,
(SELECT @cur_rank :=0, @pre_score = NULL, @pre_course_id := NULL) r
ORDER BY
course_id,
score DESC;
可以将上述的IF条件提取出来:
SELECT
s.course_id,
s.score,
IF(@pre_score = s.score, @cur_rank, @cur_rank := @cur_rank + 1) temp1,
@pre_score := s.score temp2,
IF(@pre_course_id = s.course_id, @cur_rank, @cur_rank := 1) ranking,
@pre_course_id := s.course_id
FROM
score s,
(SELECT @cur_rank :=0, @pre_score = NULL, @pre_course_id := NULL) r
ORDER BY
course_id,
score DESC;
结果如图: