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条)条记录.需要的可以参考一下. ...
随机推荐
- Bootstrap组件On和Off语法
$(document).on('click.bs.carousel.data-api','td',function(e){}) $(document).off('.carousel.data-api' ...
- PHP 转换snmp的时间格式
function convert_snmp_timeticks($timeticks) { if(!is_numeric($timeticks)) { return FALSE; } //如果是字符串 ...
- C++学习笔记12:运算符重载(赋值操作符1)
为数偶类定义专用的赋值操作符 class Couple { public: Couple(, ) :_a(a), _b(b) {} Couple(const Couple &c):_a(c._ ...
- insertAdjacentHTML
/** * insertAdjacentHTML * 支持 insertAdjacentHTML()方法的浏览器有 IE.Firefox 8+.Safari.Opera 和 Chrome */ var ...
- css在各浏览器中的兼容问题
CSS对浏览器的兼容性有时让人很头疼,或许当你了解当中的技巧跟原理,就会觉得也不是难事,从网上收集了IE7,6与Fireofx的兼容性处理方法并 整理了一下.对于web2.0的过度,请尽量用xhtml ...
- input type=file
(1)首先来说一下,如何让 <input type='file' >成为你想要的模样. 最简单的方法就是在让<input type='file' >的透明度为0(完全透明),然 ...
- Html与CSS布局技巧
一.单列布局 1.水平居中:(注:下面各个实例中实现的是child元素的对齐操作,child元素的父容器是parend元素) 1-1:使用inline-block和text-align实现: .par ...
- html5的改进与沿革
HTML5提供了一些新的元素和属性,例如<nav>(网站导航块)和<footer>.这种标签将有利于搜索引擎的索引整理,同时更好的帮助小屏幕装置和视障人士使用,除此之外,还为其 ...
- IOS中如何获取手机的当前IP
网上有许多类似的帖子.在搜索了资料以后.觉得下面的方法是最简单的. 使用的时候直接把类方法拖到自己新建的分类中就行. .h文件 #import <Foundation/Foundation.h& ...
- C++编程开发学习的50条建议(转)
每个从事C++开发的朋友相信都能给后来者一些建议,但是真正为此进行大致总结的很少.本文就给出了网上流传的对C++编程开发学习的50条建议,总结的还是相当不错的,编程学习者(不仅限于C++学习者)如果真 ...