我试图按照我之前计算过的积分对学生进行排名 但问题是,如果学生有相同的分数,他们都应该在同一级别 E.g
学生1有满分 学生2有满分
他们都必须排名为1;
我想要做的查询是(仅用于选择然后我可以将值插入到我的列中)
SELECT a.points
count(b.points)+1 as rank
FROM examresults a left join examresults b on a.points>b.points
group by a.points;
编辑以便更清楚:
他们的队伍应该像
我当前的查询返回的值如
因为缺少第三名。 (因为第二级有2个值)
答案 0 :(得分:3)
这只是使用变量修复Gordon解决方案。事情是你的排名功能不是排名应该工作的方式。 (学生4应该是4级)
SQL Fiddle Demo 您可以添加更多学生来改进测试。
select er.*,
(@rank := if(@points = points,
@rank,
if(@points := points,
@rank + 1,
@rank + 1
)
)
) as ranking
from students er cross join
(select @rank := 0, @points := -1) params
order by points desc;
<强>输出强>
| id | points | ranking |
|----|--------|---------|
| 1 | 80 | 1 |
| 2 | 78 | 2 |
| 3 | 78 | 2 |
| 4 | 77 | 3 |
| 5 | 66 | 4 |
| 6 | 66 | 4 |
| 7 | 66 | 4 |
| 8 | 15 | 5 |
答案 1 :(得分:0)
您想要一个真正的排名,它是由ANSI标准rank()
函数计算的。您可以使用以下逻辑在MySQL中实现此功能:
select er.*,
(select 1 + count(*)
from examresults er2
where er2.points > er.points
) as ranking
from exampleresults er;
对于较大的表,您可以使用变量执行此操作,但这是相当尴尬的:
select er.*,
(@rank := if(@rn := @rn + 1 -- increment row number
if(@points = points, @rank, -- do not increment rank
if(@points := points, -- set @points
@rn, @rn -- otherwise use row number
)
)
)
) as ranking
from examresults er cross join
(select @rn := 0, @rank := 0, @points := -1) params
order by points desc;
答案 2 :(得分:0)
这个查询实现了你想要的:
SELECT student_id , points, (select count(distinct(points))+1 as rank
from examresults internal
where internal.points > external.points order by points)
FROM examresults external
group by student_id