mysql巧用连表查询各科成绩前三名

下列是各表的详情,不想本身建表的同窗能够直接copy code,数据随意。 
建立表成绩详情表: sql

CREATE TABLE score ( 
id int(10) NOT NULL AUTO_INCREMENT, 
subject_id int(10) DEFAULT NULL, 
student_id int(10) DEFAULT NULL, 
score float DEFAULT NULL, 
PRIMARY KEY (id) 
) ENGINE=InnoDB AUTO_INCREMENT=19 DEFAULT CHARSET=utf8; 

这里写图片描述
 
建立学生表: spa

CREATE TABLE student ( 
id int(10) NOT NULL AUTO_INCREMENT, 
name varchar(10) DEFAULT NULL, 
PRIMARY KEY (id) 
) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8; 

学生表结构
 
建立科目表: code

CREATE TABLE subject ( 
id int(10) NOT NULL AUTO_INCREMENT, 
name varchar(10) DEFAULT NULL, 
PRIMARY KEY (id) 
) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=utf8; 

科目表结构

 

查询语句:blog

select a.id,a.subject_id,a.student_id,a.score from score as a left join score as b on a.subject_id=b.subject_id and a.score>=b.score
group by a.subject_id,a.student_id,a.score
having count(a.subject_id)>=4
order by a.subject_id,a.score desc;


分析:先将查询语句分别拆开来一步一步分析 排序

select a.id,a.subject_id,a.student_id,a.score,b.id,b.subject_id,b.student_id,b.score 
from score as a left join score as b on a.subject_id=b.subject_id;

#这里把全部的列都列出来了便于对比 
这里把表score的每一条同subject_id的数据都链接起来,造成笛卡尔积,如图所示:共18*6=108条数据 get

left join

如今咱们能够再进一步处理上面的数据了。这里咱们再加上 a.score>=b.score 这个条件筛选再进行一次筛选。 class

select a.id,a.subject_id,a.student_id,a.score,b.id,b.subject_id,b.student_id,b.score 
from score as a left join score as b on a.subject_id=b.subject_id and a.score>=b.score; 

a.score>=b.score 这里是在同一门课程中,将每个分数与其余分数(包括本身)进行一一对比,只留下大于本身,或者等于本身的分数。select

若是选择对比的行中的a.score是最高分,那么在后面利用group by a.subject_id,a.student_id,a.score分组的时候,此时计算得出的count(a.subject_id)就是最多的(count为总人数),由于其它的分数最多也只是和它同样多,其它的都比它低;同理,若是a.score是最低分,那么count(a.subject_id)是最少的(count最少为1,只有它本身,其他分数都比它高;最多为总人数,这种状况是其它人的分数都和最低分同样多...),其它的分数最差也和它同样多,其它的都比它要高。例如:float

  • 100分是最高的,因此几乎其余全部分数都符合100>=其余分数 这个条件,因此100分出现次数最多(count为总人数)
  • 0分,是最低分,几乎其余全部分数都不符合0>=其余分数这个条件,因此0分出现的次数应该是最少的(count最少为1;最多为总人数,此时其余的分数也都是最低分,即你们分数同样低)


有同窗可能会问为何不用a.score > b.score来筛选。若是用a.score > b.score来进行筛选的话,若是数据中某个科目出现大量的并列第一名的话那么第一名就会被过滤掉,以致于得不到结果。如图: im

这里写图片描述


接下来就是分组:group by a.subject_id,a.student_id,a.score #按subject_id,student_id,score来进行分组

(这里使用group by a.subject_id,a.student_id,a.score和使用group by a.subject_id,a.student_id同样的,由于两表左链接以后,不可能出现相同的a.subject_id,a.student_id有多条不一样的a.score的记录;由于同一个同窗a.student_id,同一个科目a.subject_id,只能有一个分数a.score,一个同窗不可能一个科目有多个不一样的分数); 

select a.id,a.subject_id,a.student_id,a.score,b.id,b.subject_id,b.student_id
b.score,count(a.subject_id) from score as a left join score as b 
on a.subject_id=b.subject_id and a.score>=b.score group by a.subject_id,a.student_id,a.score;

添加count(a.subject_id)来进行对比易于理解 

这里写图片描述

分组后再进行条件查询:having count(a.subject_id)>=4;

下面来讨论下>=4是什么含义:正常来讲,若是每门课程的各个同窗的分数都不同,那么同一门课程中从最高分到最低分的count(a.subject_id) 分别为:6,5,4,3,2,1;取count>=4就是取6,5,4即取count最多的三个,因此取出的数据就是排名前三(count从高到低,取前三,那么就是前三甲的记录):
接下来就是排序:order by a.subject_id,a.score desc。

这里写图片描述

 

下面有篇文章,讲解的很详细:https://www.jianshu.com/p/fff5d1f71c0f

相关文章
相关标签/搜索