Bootstrap

Mysql 常见排名实现

1.知识和数据准备

MySQL的排名函数为我们提供了强大而灵活的方式来处理数据的排序和排名,通过Mysql官网学习整理如下:

最常见的就是学校学生成绩排名,想要查看学生语文成绩排名情况,结果展示如下:

select student_id,
       subject,
       score,
       row_number() over (partition by subject order by score desc) `row_number`,
       rank() over (partition by subject order by score desc)       `rank`,
       dense_rank() over (partition by subject order by score desc) `dense_rank`
from student_subject_scroe
where subject = 'China';
+------------+---------+-------+------------+------+------------+
| student_id | subject | score | row_number | rank | dense_rank |
+------------+---------+-------+------------+------+------------+
| 01         | China   |    97 |          1 |    1 |          1 |
| 02         | China   |    96 |          2 |    2 |          2 |
| 03         | China   |    96 |          3 |    2 |          2 |
| 04         | China   |    96 |          4 |    2 |          2 |
| 05         | China   |    92 |          5 |    5 |          3 |
| 06         | China   |    90 |          6 |    6 |          4 |
+------------+---------+-------+------------+------+------------+

数据准备:

create table student_subject_scroe
(
    student_id varchar(255) comment '学生编号',
    subject    varchar(255) comment '课程名称',
    score      int comment '分数'
);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('01', 'english', 89);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('01', 'math', 61);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('01', 'china', 97);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('02', 'english', 87);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('02', 'math', 53);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('02', 'china', 96);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('03', 'english', 87);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('03', 'math', 53);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('03', 'china', 96);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('04', 'english', 84);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('04', 'math', 52);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('04', 'china', 96);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('05', 'english', 74);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('05', 'math', 47);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('05', 'china', 92);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('06', 'english', 73);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('06', 'math', 40);
insert into dwd.student_subject_scroe (student_id, subject, score) values ('06', 'china', 90);

2.顺序排序,不重复不间断的连续序号

需求:想要查看每门学科对应的成绩排名,排名不重复不间断

-- 使用row_number()函数实现
select student_id,
       subject,
       score,
       row_number() over (partition by subject order by score desc) `row_number`
from student_subject_scroe;
-- 通过定义变量实现
select student_id,
       subject,
       score,
       `row_number`
from (select a.student_id,
             a.subject,
             a.score,
             if(@pre_subject = a.subject,   -- 排序后,判断上一行的学科是否等于当前行
				@row_num := @row_num + 1,   -- 是,返回@row_num+1,并且再次赋值给该变量@row_num
				@row_num := 1) `row_number`,-- 否,重置@row_num变量为1
             @pre_subject := a.subject      -- 将当前行的学科赋值给@pre_subject
      from student_subject_scroe as a,
           (select @pre_subject := null, @row_num := 0) b -- 临时表b,定义初始变量
      order by subject, score desc  -- 根据学科,分数降序排
) as c;

row_number()使用直接根据学科分组,分数降序即可得到结果;

通过定义变量实现,可以理解为select是一行行记录查询出来的,弄清变量的赋值的先后顺序,每行该变量的值,那么就清楚了。

3.并列排名,会跳过重复的序号

需求:想要查看每门学科对应的成绩排名,排名会重复间断

-- 使用rank()函数实现
select student_id,
       subject,
       score,
       rank() over (partition by subject order by score desc)       `rank`
from student_subject_scroe;
-- 通过定义变量实现
select student_id,
       subject,
       score,
       `rank`
from (select a.student_id,
             a.subject,
             a.score,
             IF(@pre_subject = a.subject,     -- 排序后,判断上一行的学科是否等于当前行
                @row_num := @row_num + 1,     -- 是,返回@row_num+1,并且再次赋值给该变量@row_num
                @row_num := 1) `row_number`, -- 否,重置@row_num变量为1
             IF(@pre_subject = a.subject,     -- 排序后,判断上一行的学科是否等于当前行
                IF(@pre_score = a.score,      -- 上一行学科等于当前行,则判断上一行分数是否等于当前行分数
				   @cur_rank,                 -- 分数相等,则输出@cur_rank
				   @cur_rank := @row_num),  -- 分数不等,则输出`row_number`的@row_num,注意要赋值给@cur_rank,给下一行引用
                @cur_rank := 1) `rank`,       -- 学科不等,则重置@cur_rank变量为1
             @pre_subject := a.subject,       -- 将当前行的学科赋值给@pre_subject
             @pre_score := a.score            -- 将当前行的分数赋值给@pre_score
      from student_subject_scroe as a,
           (select @pre_subject := null,@pre_score := null,@cur_rank := 0,@row_num := 0) b -- 临时表b,定义初始变量
      order by subject, score desc) as c -- 根据学科,分数降序排
;     

rank()使用直接根据学科分组,分数降序即可得到结果;

通过定义变量实现,借助根据步骤2中顺序排名,不重复跳跃的序号的变量实现,弄清变量的赋值的先后顺序,每行该变量的值,那么也就处理了。

4.并列排序,有重复间断的序号

需求:想要查看每门学科对应的成绩排名,排名重复不间断

-- 使用dense_rank()函数实现
select student_id,
       subject,
       score,
       dense_rank() over (partition by subject order by score desc) `dense_rank`
from student_subject_scroe;
-- 通过定义变量实现
select student_id,
       subject,
       score,
       `rank`
from (select a.student_id,
             a.subject,
             a.score,
             IF(@pre_subject = a.subject,      -- 排序后,判断上一行的学科是否等于当前行
                IF(@pre_score = a.score,       -- 上一行学科等于当前行,则判断上一行分数是否等于当前行分数
				   @cur_rank,                  -- 分数相等,则输出@cur_rank
				   @cur_rank := @cur_rank + 1),-- 分数不等,则输出@cur_rank+1,注意要赋值给@cur_rank,给下一行引用
                @cur_rank := 1) `rank`,
             @pre_subject := a.subject,        -- 将当前行的学科赋值给@pre_subject
             @pre_score := a.score             -- 将当前行的分数赋值给@pre_score
      from student_subject_scroe as a,
           (select @pre_subject := null,@pre_score := null,@cur_rank := 0) b
      order by subject, score desc) as c       -- 根据学科,分数降序排
;

dense_rank()使用直接根据学科分组,分数降序即可得到结果;

通过定义变量实现,理解了步骤2和步骤3的排名实现,这也就简单了。

可以发现用排序函数特别简单,Mysql 8.0 之后的版本有窗口函数,之前的版本没有,所以赶紧下载新版本用起来。

悦读

道可道,非常道;名可名,非常名。 无名,天地之始,有名,万物之母。 故常无欲,以观其妙,常有欲,以观其徼。 此两者,同出而异名,同谓之玄,玄之又玄,众妙之门。

;