美文网首页MySQL数据库
SQL分组查询找组内最大

SQL分组查询找组内最大

作者: SpaceCat | 来源:发表于2019-09-29 09:06 被阅读0次

1、问题

现在有数据表,存储着各个学生的各个科目的成绩。如果查出每门课程得分最高的学生该怎么找呢?
正常的思路可能是先按分组查出每个科目的最高成绩,然后再将这个结果和原表做一个连接查询(按照科目和成绩相等连接),这样才可以查出想要的结果。或者是,多写几个子查询,应该也能解决问题。
但是,这些答案都不好,没有充分发挥SQL语法中关系代数的优势。本文主要讲如何通过自连接,解决这个问题。

2、准备

2.1 表结构

建立如下表结构:

create table Grade (
    StudentName varchar2(50),
    CourseName varchar2(100),
    Grade number
);

这里省略了主键和索引等。

2.2 数据

执行如下脚本,向表中插入数据:

INSERT INTO grade (StudentName, CourseName, Grade)
VALUES ('John Smith', 'English', 76);

INSERT INTO grade (StudentName, CourseName, Grade)
VALUES ('John Wall', 'Math', 86);

INSERT INTO grade (StudentName, CourseName, Grade)
VALUES ('Kobe Byrant', 'Math', 90);

INSERT INTO grade (StudentName, CourseName, Grade)
VALUES ('Will Smith', 'English', 96);

INSERT INTO grade (StudentName, CourseName, Grade)
VALUES ('Li Lei', 'Math', 99);


INSERT INTO grade (StudentName, CourseName, Grade)
VALUES ('Tim Duncan', 'English', 95);

COMMIT;

数据如下:

STUDENTNAME COURSENAME GRADE
John Smith English 76
John Wall Math 86
Kobe Byrant Math 90
Will Smith English 96
Li Lei Math 99
Tim Duncan English 95

3、查询语句

3.1 思路

这里的解决方案是做一个自连接(连接条件:t.CourseName = tt.CourseName AND tt.grade >= t.grade),查找比当前记录分数高的记录只有一条的记录(这一条自然就是记录本身了)。自连接之后的筛选方法是,先按照t.StudentName, t.CourseName分组,然后,筛选出只有一条记录的组就好了。

3.2 自连接SQL v1.0

根据上面的思路,可以写出如下的SQL:

SELECT t.StudentName,
       t.CourseName
FROM Grade t,
     Grade tt
WHERE t.CourseName = tt.CourseName
  AND tt.grade >= t.grade
GROUP BY t.StudentName,
         t.CourseName
HAVING count(1) <= 1

结果如下:

STUDENTNAME COURSENAME
Will Smith English
Li Lei Math

3.3 自连接SQL v2.0

前面的结果中,没法看到各个科目成绩最高的学生的分数,因为分数字段没有出现在group by表达式中。
这个非常好解决,不会有相同StudentName和CourseName,但是成绩不同的情况,所以,直接在group by子句中加上这个字段就行。如下。

SELECT t.CourseName,
       t.StudentName,
       t.grade
FROM Grade t,
     Grade tt
WHERE t.CourseName = tt.CourseName
  AND tt.grade >= t.grade
GROUP BY t.StudentName,
         t.CourseName,
         t.grade
HAVING count(1) <= 1

查询结果如下:

COURSENAME STUDENTNAME GRADE
English Will Smith 96
Math Li Lei 99

3.4 查询各个科目的成绩排名

根据这个原理,可以很容易地查询各个科目的成绩排名。
自连接之后,有多少条比当前记录大的(实际是大于等于),就认为排名是第几。

SELECT t.CourseName,
       t.StudentName,
       t.grade,
       count(1) CourseRank
FROM Grade t,
     Grade tt
WHERE t.CourseName = tt.CourseName
  AND tt.grade >= t.grade
GROUP BY t.StudentName,
         t.CourseName,
         t.grade
order by t.CourseName, CourseRank

结果如下:

COURSENAME STUDENTNAME GRADE COURSERANK
English Will Smith 96 1
English Tim Duncan 95 2
English John Smith 76 3
Math Li Lei 99 1
Math Kobe Byrant 90 2
Math John Wall 86 3

4、说明

这个问题,oracle中应该有专门的partition by语法专门解决类似问题,MySql中应该也有GROUP_CONCAT聚合函数能够配合SUBSTRING_INDEX截取函数可以实现类似功能。但是,这些高级的语法有一个局限性,他们不通用。了解这种自连接的思路可以帮助我们更好地理解SQL的强大语法特性。

相关文章

  • SQL分组查询找组内最大

    1、问题 现在有数据表,存储着各个学生的各个科目的成绩。如果查出每门课程得分最高的学生该怎么找呢?正常的思路可能是...

  • mysql在测试中的应用2

    sql 语法 聚合函数及分组筛选: 在一组数据中查询:总计数量count‘合计sum,平均值avg,最大max,最...

  • Sql分组后查询条数

    先看下一般的查询sql语句和分组sql语句 但是想知道分组后,一共有多少组数据怎么做呢,下面我们用子查询的方式实现...

  • EF Core 备忘

    模糊查询sql linq 内连接查询sql linq 左连接查询sql linq 左连接查询(连接内带条件)sql...

  • mysql中sql_mode=only_full_group_b

    背景:这是一个菜单表,类型有目录、菜单、按钮。需求:根据类型分组查看组内最大的id和菜单标题。 问题:sql_mo...

  • SQL分组查询

    一.分组查询 1.使用group by进行分组查询 在使用group by关键字时,在select列表中可以指定的...

  • 面试

    面试 问逻辑题 挺简单的 网上可以搜,逻辑题,写sql 表内查询分组查询之类的,问一些其他问题 面试官很nice,...

  • SQL进阶部分一

    分组查询 什么是分组查询 ​ 将查询结果按照1个或多个字段进行分组,字段值相同的为一组 分组使用 ​ ...

  • sql 分组查询-mysql

    group by <分组项> [having <分组筛选条件>] 1.统计各班人数: select 班级编号,co...

  • SQL语句:分组查询

    在对数据表中数据进行统计时,可能需要按照一定的类别分别进行统计。比如查询每个部门的员工数。 使用GROUP BY按...

网友评论

    本文标题:SQL分组查询找组内最大

    本文链接:https://www.haomeiwen.com/subject/ymnxpctx.html