mysql 获取行号,分数相等行号也相等

本文介绍如何使用MySQL查询实现分数排名,当分数相同时,排名也相同,保持排名连续。通过示例展示了具体SQL语句的实现原理和逻辑。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1 获取行号

编写一个 SQL 查询来实现分数排名。如果两个分数相同,则两个分数排名(Rank)相同。请注意,平分后的下一个名次应该是下一个连续的整数值。换句话说,名次之间不应该有“间隔”。

+----+-------+
| Id | Score |
+----+-------+
| 1  | 3.50  |
| 2  | 3.65  |
| 3  | 4.00  |
| 4  | 3.85  |
| 5  | 4.00  |
| 6  | 3.65  |
+----+-------+
例如,根据上述给定的 Scores 表,你的查询应该返回(按分数从高到低排列):

+-------+------+
| Score | Rank |
+-------+------+
| 4.00  | 1    |
| 4.00  | 1    |
| 3.85  | 2    |
| 3.65  | 3    |
| 3.65  | 3    |
| 3.50  | 4    |
+-------+------+

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/rank-scores
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

select Score,cast(@rownum :=@rownum + (@pre <> (@pre :=Score)) as signed) as Rank
from Scores,(select @rownum :=0) a ,(select @pre :=-1) b
order by Score desc

解释

整理了一下几个不好理解的点:

  1. @a 类似在 Oracle 中的rownum,可以在生成结果内附加上一列序列号,可以近似理解为查询后加上行号;

  2. @a := @a + 1 实际上是赋值,旧值+1变为新值赋给@a;

  3. 实际上并没有直接 @a+1 那么简单,还要先去判断分数是否与前一行相同,所以引入 @pre 来记录:

    • 先将Score赋值给: @pre := Score;
    • 然后判断之前的 @pre 是否与赋值后的 @pre 不相同 ==> (@pre <> (@pre := Score)) “<>” 就是 "!=" 的意思;
    • 两者不同,判断结果为真,则取1; 两者相同,判断结果为假,取0, 最后再用 0或1 加上 @a 即为当前行分数的排名;
  4. (select @a := 0, @pre := -1) t 为初始化 @a 和 @pre 的开始值;

    • @pre 初始值为 -1 为的是防止Score有 0 分的出现;
    • @第一次比较Score值的时候,@pre初始值肯定和Score不同, 所以第一次比较 @a 必然会 +1,所以@a要从0开始;
  5. 最后以将结果根据Score进行倒序展示;

以上解释内容来自leetcode

https://leetcode-cn.com/problems/rank-scores/solution/shi-yong-zi-bian-liang-de-jie-fa-jian-dan-yi-dong-/

cast好用,round,floor,并不行不知道为什么

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值