今天遇到一个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 分组排名的更多相关文章

  1. Mysql 分组选择

    Mysql 分组选择 在其他的数据库中我们遇到分组选择的问题时,比如在分组中计算前10名的平均分 我们可以使用row_number()over() 比较方便的得到. 但是在mysql中,问题就被抛了出 ...

  2. mysql成绩排名

    关于mysql成绩排名,网上大部分只是order by简单排序,忽略了成绩相同并列名次的问题. 定义了一个表score结构为:

  3. Oracle和MySQL分组查询GROUP BY

    Oracle和MySQL分组查询GROUP BY 真题1.Oracle和MySQL中的分组(GROUP BY)有什么区别? 答案:Oracle对于GROUP BY是严格的,所有要SELECT出来的字段 ...

  4. mysql 分组和聚合函数

    mysql 分组和聚合函数 Mysql 聚集函数有5个: 1.COUNT() 记录个数(count(1),count(*)统计表中行数,count(列名)统计列中非null数) 2.MAX() 最大值 ...

  5. mysql 分组内 排序

    mysql 分组内 排序 类似于 sqlserver over partition by   因为mysql中木有sqlserver over partition by这个函数,要从sqlserver ...

  6. mysql计算排名 转

    from :http://www.cnblogs.com/aeiou/p/5719396.html http://www.cnblogs.com/zengguowang/p/5541431.html ...

  7. MySQL实现排名并查询指定用户排名功能,并列排名功能

    MySQL实现排名并查询指定用户排名功能,并列排名功能 表结构: CREATE TABLE test.testsort ( id int(11) NOT NULL AUTO_INCREMENT, ui ...

  8. mysql计算排名

    mysql计算排名,获取行号rowno 学生成绩表数据 SELECT * FROM table_score ORDER BY score DESC; 获取某个学生成绩排名并计算该学生和上一名学生成绩差 ...

  9. mysql分组取最大(最小、最新、前N条)条记录

    在数据库开发过程中,我们要为每种类型的数据取出前几条记录,或者是取最新.最小.最大等等,这个该如何实现呢,本文章向大家介绍如何实现mysql分组取最大(最小.最新.前N条)条记录.需要的可以参考一下. ...

随机推荐

  1. 自定义Session类实现

    public class CustomizeSession { private HttpSessionState _sesssion = HttpContext.Current.Session; pu ...

  2. Unity3D ShaderLab 立方体图的菲涅尔反射

    Unity3D ShaderLab 立方体图的菲涅尔反射 菲涅尔反射是反射类型中比较常见的一种类型,当我们的视线正对物体表面,那么反射量会明显增加, 我们几乎可以在任何支持反射类型的物体表面看到这种情 ...

  3. [转]iOS游戏如何防御外挂及IAP破解

    http://www.j1f3.com/news/game/21371.html 今年3月初写过一篇<iO平台游戏安全小议>,到现今已有7个月了.在这段时间内,iOS平台上的安全问题也产生 ...

  4. 转:C#精髓 第四讲 GridView 72般绝技

    说明:准备出一个系列,所谓精髓讲C#语言要点.这个系列没有先后顺序,不过尽量做到精.可能会不断增删整理,本系列最原始出处是csdn博客,谢谢关注. C#精髓 第四讲 GridView 72般绝技 作者 ...

  5. druid 源码分析与学习(含详细监控设计思路的彩蛋)(转)

    原文路径:http://herman-liu76.iteye.com/blog/2308563  Druid是阿里巴巴公司的数据库连接池工具,昨天突然想学习一下阿里的druid源码,于是下载下来分析了 ...

  6. 【题解】【DP】【Leetcode】Climbing Stairs

    You are climbing a stair case. It takes n steps to reach to the top. Each time you can either climb ...

  7. Canvas 获取颜色值

    Canvas 是 HTML5 的画布元素,按照像素绘制图像.有时需要用户点击鼠标的时候获取像素值. 获取画布元素 var canvas = document.getElementById(" ...

  8. [翻译]Java垃圾收集精粹(Java Garbage Collection Distilled)

    source URL: http://www.infoq.com/articles/Java_Garbage_Collection_Distilled Name: Java Garbage Colle ...

  9. 几个有用的SAP安全配置的用户参数配置列表

    转自http://blog.sina.com.cn/s/blog_4f913cf80100mksj.html Parameter Brief Description login/min_passwor ...

  10. Android SDK Manager Google Apis 下载

    本意是想利用google的gcm来实装android推送功能的,很遗憾, google貌似已经停止提供啥服务给国内了,或者说国内想继续使用google 服务暂时变得几乎不可能了.找了个代理来进行goo ...