MYSQL 分组排名
今天遇到一个MYSQL排序的问题,要求按某列进行分组,组内进行排序.
百度一下发现MYSQL不支持row_number(),rank()等函数.
采用的办法如下,我们首先创建一个测试表:
--创建表
create table Rank_test(ID int,SCORE int,grp int) --插入数据
insert into Rank_test values(1 , 28,1);
insert into Rank_test values(2 , 33,1);
insert into Rank_test values(3 , 33,1);
insert into Rank_test values(4 , 89,1);
insert into Rank_test values(5 , 99,1);
insert into Rank_test values(6 , 68,1);
insert into Rank_test values(7 , 68,1);
insert into Rank_test values(8 , 78,1);
insert into Rank_test values(9 , 88,1);
insert into Rank_test values(10, 90 ,1);
insert into Rank_test values(11, 28,2);
insert into Rank_test values(12, 33,2);
insert into Rank_test values(13, 33,2);
insert into Rank_test values(14, 89,2);
insert into Rank_test values(15, 99,2);
insert into Rank_test values(16, 68,2);
insert into Rank_test values(17, 68,2);
insert into Rank_test values(18, 78,2);
insert into Rank_test values(19, 88,2);
insert into Rank_test values(20, 90 ,2);
分组排序的SQL代码:
SELECT id,
score,
rank
FROM (SELECT tmp.id,
tmp.score,
@rank := (case when @Grp = grp then @rank + 1 else 1 end) AS rank ,
@Grp:=grp as grp2
FROM (SELECT id,
score,
grp
FROM rank_test) tmp,
(SELECT @rank := 0,@grp:=0) a
order by grp,score desc) RESULT
;
结果如下:

测试结果OK,但重点是套用到我们的SQL中后,排序结果完全不是我们预期中的,而且没有找到规律.
经过几次尝试,最后重新改写了SQL代码,终于成功了.
原因是套用下面这段代码时,我们是用了几张表进行关联(inner join ),最后将inner join 的表全部改为where 语句中的子句,结果满足我们的预期.
(SELECT id,
score,
grp
FROM rank_test) tmp,
所以建议在这里的代码尽量使用单表,不要多张表关联
附上修改前后的代码:(红色部分是修改前,绿色部分是修改后)
select * from (
select evtdate
,stkabb,chng,PCTCSHG,cshg,SHHNAME
,@rank := (case when @SHHNAME=SHHNAME then @rank + 1
else 1 end) AS rank
,@SHHNAME:=SHHNAME
from
(
select tt.SHHNAME -- 股东名称
,s.evtdate -- 变动日期
,r.stkabb -- 变动股票名称
,ifnull(s.RLDSHG,s.cshg-s.SHGBCH)/10000 as chng -- 变动股数(万股)
,s.cshg/10000 as cshg-- 变动后持股(万股)
,s.PCTCSHG -- 变动后持股比例%
from pgenius.hk_stkcode r
inner join pgenius.HK_SAKCHMJSHH s
on r.comunic = s.comunic
inner join
(select distinct b.SHHNAME
from pgenius.hk_stkcode a
inner join pgenius.HK_SAKCHMJSHH b
on a.comunic = b.comunic
and b.evtdate = (select max(evtdate) from pgenius.HK_SAKCHMJSHH
where comunic = b.comunic and SHHNAME = b.SHHNAME and NTUREFCINTRTS = b.NTUREFCINTRTS)
where a.lssturefc = 1
and b.cshg > 0
and a.stkcode = ''
) tt
on s.SHHNAME = tt.SHHNAME
) t1,
(SELECT @rank := 0,@SHHNAME:='') t2
order by SHHNAME,evtdate desc,stkabb ) t
select t.*
, o.stkabb
from
(
select t1.*
,@rank := (case when @SHHNAME=SHHNAME then @rank + 1
else 1 end) AS rank
,@SHHNAME:=SHHNAME
from
(
select s.evtdate
,ifnull(s.RLDSHG,s.cshg-s.SHGBCH)/10000 as chng -- 变动股数(万股)
,s.cshg/10000 as cshg-- 变动后持股(万股)
,s.PCTCSHG -- 变动后持股比例%
,SHHNAME
,s.COMUNIC
from pgenius.HK_SAKCHMJSHH s
where SHHNAME in (
select distinct shhname from pgenius.HK_SAKCHMJSHH a
where comunic=(select COMUNIC from pgenius.hk_stkcode where stkcode='')
and ISVALID=1
and not exists(select 1 from pgenius.HK_SAKCHMJSHH where COMUNIC=a.COMUNIC and SHHNAME=a.SHHNAME and evtdate>a.evtdate)
and cshg > 0)
) t1,
(SELECT @rank := 0,@SHHNAME:='') t2
order by SHHNAME,evtdate desc
) t
inner join pgenius.hk_stkcode o on t.COMUNIC= o.COMUNIC
where t.rank<=10
order by shhname,rank
MYSQL 分组排名的更多相关文章
- Mysql 分组选择
Mysql 分组选择 在其他的数据库中我们遇到分组选择的问题时,比如在分组中计算前10名的平均分 我们可以使用row_number()over() 比较方便的得到. 但是在mysql中,问题就被抛了出 ...
- mysql成绩排名
关于mysql成绩排名,网上大部分只是order by简单排序,忽略了成绩相同并列名次的问题. 定义了一个表score结构为:
- Oracle和MySQL分组查询GROUP BY
Oracle和MySQL分组查询GROUP BY 真题1.Oracle和MySQL中的分组(GROUP BY)有什么区别? 答案:Oracle对于GROUP BY是严格的,所有要SELECT出来的字段 ...
- mysql 分组和聚合函数
mysql 分组和聚合函数 Mysql 聚集函数有5个: 1.COUNT() 记录个数(count(1),count(*)统计表中行数,count(列名)统计列中非null数) 2.MAX() 最大值 ...
- mysql 分组内 排序
mysql 分组内 排序 类似于 sqlserver over partition by 因为mysql中木有sqlserver over partition by这个函数,要从sqlserver ...
- mysql计算排名 转
from :http://www.cnblogs.com/aeiou/p/5719396.html http://www.cnblogs.com/zengguowang/p/5541431.html ...
- MySQL实现排名并查询指定用户排名功能,并列排名功能
MySQL实现排名并查询指定用户排名功能,并列排名功能 表结构: CREATE TABLE test.testsort ( id int(11) NOT NULL AUTO_INCREMENT, ui ...
- mysql计算排名
mysql计算排名,获取行号rowno 学生成绩表数据 SELECT * FROM table_score ORDER BY score DESC; 获取某个学生成绩排名并计算该学生和上一名学生成绩差 ...
- mysql分组取最大(最小、最新、前N条)条记录
在数据库开发过程中,我们要为每种类型的数据取出前几条记录,或者是取最新.最小.最大等等,这个该如何实现呢,本文章向大家介绍如何实现mysql分组取最大(最小.最新.前N条)条记录.需要的可以参考一下. ...
随机推荐
- LCA(倍增)
type arr=record v,nt:longint; end; ; lx=; ..maxn] of longint; eg:..maxn*] of arr; d:..maxn] of longi ...
- 基于tcpdump实例讲解TCP/IP协议
前言 虽然网络编程的socket大家很多都会操作,但是很多还是不熟悉socket编程中,底层TCP/IP协议的交互过程,本文会一个简单的客户端程序和服务端程序的交互过程,使用tcpdump抓包,实例讲 ...
- jconsole远程查看jvm性能
Java VisualVM -- JDK自带的强大性能监测工具 这里给出了另一种,可以通过jdk下bin目录下的jconsole.exe来观察java程序内存的使用及变化情况 一般远程调试可以用,性能 ...
- Linux主机硬盘的主要规划
系统对于硬盘的需求跟刚刚提到的主机开放的服务有关,那么除了这点之外,还有没有其他的注意事项?当然有,那就是数据的分类与数据安全性的考虑,即当主机系统的硬件出现问题时,你的档案数据能否安全的保存. 前篇 ...
- TNF-mutithread 编译过程记录
地址 https://github.com/msng4t/TNF-mutithread 地址 https://github.com/msng4t/TNF 目的:想要寻找一套性能,可读性相对较好的Soc ...
- ES6—解构赋值
1.什么是解构赋值 ES6允许按照预定的模式,从数组.对象中提取值,对变量进行赋值. 我们直接用例子说明. 2. 数组的解构赋值 数组传统的变量赋值: var arr=[1,2,3]; ...
- Kernel panic - not syncing: Attempted to kill init
解决方法:系统启动的时候,按下‘e’键进入grub编辑界面,编辑grub菜单,选择“kernel /vmlinuz-2.6.23.1-42.fc8 ro root=/dev/vogroup00/log ...
- 《笨办法学Python》
习题一 第一个程序 print "Hello World!" print "Hello Evilxr" print "I like typing th ...
- kuangbin_ShortPath M (POJ 1062)
提出了一个错误的算法 以为能优化到只运行两次dij 然而我还是too naive 还是乖乖dij n 次吧... #include <iostream> #include <stri ...
- 磁盘参数修订[转自vbird]
某些时刻,你可能会希望修改一下目前文件系统的一些相关信息,举例来说,你可能要修改 Label name , 或者是 journal 的参数,或者是其他硬盘运行时的相关参数 (例如 DMA 启动与否-) ...