SQL之分组排序取top n
转自:http://blog.csdn.net/wguangliang/article/details/50167283
要求:按照课程分组,查找每个课程最高的两个成绩。
数据文件如下:
第一列no为学号,第二列course为课程,第三列score为分数
[plain] view plain copy
1. mysql> select * from lesson;
2. +-------+---------+-------+
3. | no | course | score |
4. +-------+---------+-------+
5. | N0101 | Marth | 100 |
6. | N0102 | English | 12 |
7. | N0102 | Chinese | 55 |
8. | N0102 | History | 58 |
9. | N0102 | Marth | 25 |
10. | N0103 | English | 100 |
11. | N0103 | Chinese | 87 |
12. | N0103 | History | 88 |
13. | N0103 | Marth | 72 |
14. | N0104 | English | 20 |
15. | N0104 | Chinese | 60 |
16. | N0104 | History | 88 |
17. | N0104 | Marth | 56 |
18. | N0105 | English | 56 |
19. | N0105 | Chinese | 88 |
20. | N0105 | History | 88 |
21. | N0201 | English | 66 |
22. | N0201 | Chinese | 77 |
23. | N0201 | History | 80 |
24. | N0201 | Marth | 100 |
25. | N0202 | English | 35 |
26. | N0202 | Chinese | 56 |
27. | N0202 | History | 86 |
28. | N0202 | Marth | 99 |
29. | N0203 | English | 100 |
30. | N0203 | Chinese | 87 |
31. | N0203 | History | 88 |
32. | N0203 | Marth | 57 |
33. | N0204 | English | 98 |
34. | N0204 | Chinese | 100 |
35. | N0204 | History | 66 |
36. | N0204 | Marth | 71 |
37. | N0205 | English | 98 |
38. | N0205 | Chinese | 100 |
39. | N0205 | History | 66 |
40. | N0205 | Marth | 71 |
41. | N0301 | English | 66 |
42. | N0301 | Chinese | 89 |
43. | N0301 | History | 68 |
44. | N0301 | Marth | 83 |
45. | N0302 | English | 76 |
46. | N0302 | Chinese | 99 |
47. | N0302 | History | 80 |
48. | N0302 | Marth | 74 |
49. | N0303 | English | 100 |
50. | N0303 | Chinese | 100 |
51. | N0303 | History | 88 |
52. | N0303 | Marth | 57 |
53. | N0304 | English | 76 |
54. | N0304 | Chinese | 100 |
55. | N0304 | History | 66 |
56. | N0304 | Marth | 86 |
57. | N0305 | English | 98 |
58. | N0305 | Chinese | 100 |
59. | N0305 | History | 40 |
60. | N0305 | Marth | 59 |
61. | N0306 | English | 52 |
62. | N0306 | Chinese | 87 |
63. | N0306 | History | 72 |
64. | N0306 | Marth | 71 |
65. | N0101 | Chinese | 55 |
66. | N0101 | History | 84 |
67. | N0101 | English | 82 |
68. | N0101 | English | 82 |
69. +-------+---------+-------+
70. 64 rows in set
在hive上查询
1. select a.course,a.score
2. from
3. (
4. select course,score,row_number() over(partition by course order by score desc) as n
5. from lesson
6. )a
7. where a.n<=2;
其中:
1. row_number() over(partition by course order by score desc)
意思是以课程分组,按成绩递减排序,并为每组中的数据打上行号的标记,从1开始。
这样,再在外层套一层过滤行号小于等于2的即可:-D
查询结果如下图1所示:

图1 Hive查询结果
在mysql上查询
由于MySQL不支持row_number()over()等窗口函数
方法1.自查询比较
1. select course,score
2. from lesson a
3. where 2 >
4. (
5. select count(1)
6. from lesson b
7. where a.score<b.score and a.course=b.course
8. )
9. order by a.course,a.score desc;
因为是查询最高的两个成绩,所以是2>,如果查询最高的前N个成绩,改成 N>
该条sql语句的大概思路是:
从a表中拿出一条数据,与b表中所有与该条数据相同course的数据比较,统计出b表有多少相同课程的score比该条数据的score高;
如果b表中有0条比该条数据高,则该条数据是该门课程的最高分;
如果统计出有1条数据,则该条数据是该门课程分数的第二高;
但是,还存在一些问题:
比如,最高分存在多个,则会统计出多于2条的数据,如下图2统计结果也有所反应:

图2 mysql查询结果
方法2.动态sql
1. SET @row=0;
2. SET @groupid='';
3. select a.course,a.score
4. from
5. (
6. select no,course,score,case when @groupid=course then @row:=@row+1 else @row:=1 end rownum,@groupid:=course from lesson
7. order by course,score desc
8. )a
9. where a.rownum<=2;
其中:
@row用于统计行号,@groupid用于分组,记录该组的名称
1. select no,course,score,case when @groupid=course then @row:=@row+1 else @row:=1 end rownum,@groupid:=course from lesson
2. order by course,score desc
意思是:按照分组名course和需要的排序score递增 进行排序,这样,相同课程就会排在一起,且相同的课程之间按照成绩排序。
取出一条数据,如果该条数据的course与@group相同,则意味着是相同课程之间的比较,那么@row自加1。
否则意味着该条数据是另一门课程的第一条数据,则@row=1
这样每个课程就能够按照成绩排序并标记上行号
那么外层只需要过滤rownum<=2即可得到每门课的前2个最高分。
最后执行结果与hive一致,不再上图片了。
SQL之分组排序取top n的更多相关文章
- SQL Server 分组后取Top N
SQL Server 分组后取Top N(转) 近日,工作中突遇一需求:将一数据表分组,而后取出每组内按一定规则排列的前N条数据.乍想来,这本是寻常查询,无甚难处.可提笔写来,终究是困住了笔者好一会儿 ...
- sql分组排序取top
写法1: use anypay; select tr.* from (select task_code, max(created_at) as cal from task_log group by t ...
- hive分组排序 取top N
pig可以轻松获取TOP n.书上有例子 hive中比较麻烦,没有直接实现的函数,可以写udf实现.还有个比较简单的实现方法: 用row_number,生成排名序列号.然后外部分组后按这个序列号多虑, ...
- SQL语句分组排序,多表关联排序
SQL语句分组排序,多表关联排序总结几种常见的方法: 案例一: 在查询结果中按人数降序排列,若人数相同,则按课程号升序排列? 分析:单个表内的多个字段排序,一般可以直接用逗号分割实现. select ...
- 一条Sql语句分组排序并且限制显示的数据条数
如果我想得到这样一个结果集:分组排序,并且每组限定记录集的数量,用一条SQL语句能办到吗? 比如说,我想找出学生期末考试中,每科的前3名,并按成绩排序,只用一条SQL语句,该怎么写? 表[TScore ...
- mysql 分组排序取最值
查各个用户下单最早的一条记录 查各个用户下单最早的前两条记录 查各个用户第二次下单的记录 一.建表填数据: SET NAMES utf8mb4; -- 取消外键约束 ; -- ------------ ...
- mysql分组排序取最大值所在行,类似hive中row_number() over partition by
如下图, 计划实现 :按照 parent_code 分组, 取组中code最大值所在的整条记录,如红色部分.(类似hive中: row_number() over(partition by)) sel ...
- 记一次有意思的 SQL 实现 → 分组后取每组的第一条记录
开心一刻 今天,朋友气冲冲的走到我面前 朋友:我不是谈了个女朋友,谈了三个月嘛,昨天我偷看她手机,你猜她给我备注什么 我:备注什么? 朋友:舔狗 2 号! 我一听,气就上来了,说道:走,找她去,这婆娘 ...
- SQL获取分组后取某字段最大一条记录(求每个类别中最大的值的列表)
获取分组后取某字段最大一条记录 方法一:(效率最高) select * from test as a where typeindex = (select max(b.typeindex) from t ...
随机推荐
- C# 版本的24点实现
C# 版本的24点实现. 已经实现基本功能,可以正确的算 3, 3, 8, 8 这类组合. 稍加修改就可以支持任意数目的操作数和操作符组合形成的四则运算表达式,不限于24点. 代码还比较简单粗糙,晚一 ...
- 译: 5. RabbitMQ Spring AMQP 之 Topic 主题
在上一个教程中,我们提高了消息传递的灵活 我们使用direct交换而不是使用仅能够进行虚拟广播的fanout交换, 并且获得了基于路由key 有选择地接收消息的可能性. 虽然使用direct 交换改进 ...
- python thrift使用实例
前言 Apache Thrift 是 Facebook 实现的一种高效的.支持多种编程语言的远程服务调用的框架.本文将从 Python开发人员角度简单介绍 Apache Thrift 的架构.开发和使 ...
- supervisor详解
1.什么是supervisor supervisor是用python写的一个进程管理工具,用来启动,重启,关闭进程. 2.supervisor的安装 pip install supervisor 3. ...
- batch,iteration,epoch 什么意思
深度学习中经常看到epoch. iteration和batchsize,下面按自己的理解说说这三个的区别: (1)batchsize:批大小.在深度学习中,一般采用SGD训练,即每次训练在训练集中取b ...
- 11款最轻量级的CSS框架
日子一去不复返了HTML用于造型的网页.今天,CSS规则,很难想象没有它的任何网页设计.CSS在最近非常先进,用于创建复杂的Web设计和风格.那么,我们为什么要使用CSS框架?答案很简单.CSS框架主 ...
- java框架篇---Struts2 本地化/国际化(i18n)
国际化(i18n)是规划和实施的产品和服务,使他们能很容易地适应特定的本地语言和文化的过程中,这个过程被称为本地化.国际化的过程有时也被称为翻译或本地化启用.国际化是缩写i18n,因为我和两端用n字打 ...
- 完美解决"Encountered an NTFS Volume with a logfile ..."
完美解决Ghost镜像文件时出现"Encountered an NTFS Volume with a logfile that has not been flushed(536)" ...
- 【Python】socket模块应用
[Socket] 本文记录了一些socket模块的简单应用,对于具体原理还没来得及深究. ■ 利用socket模块进行端口连接验证和扫描 在linux中常用nc命令来进行远端端口是否开放的验证.但是这 ...
- 第三百九十六节,Django+Xadmin打造上线标准的在线教育平台—其他插件使用说,自定义列表页上传插件
第三百九十六节,Django+Xadmin打造上线标准的在线教育平台—其他插件使用说,自定义列表页上传插件 设置后台列表页面字段统计 在当前APP里的adminx.py文件里的数据表管理器里设置 ag ...