LeetCode:178.分数排名
题目
编写一个 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
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
解答
使用 oracle 或者 hive 的开窗函数,直接实现。
---- oracle ----
/* Write your PL/SQL query statement below */
select Score,
dense_rank() over(order by Score desc) as Rank
from Scores; ---- 473ms
这么简单,不应该是中等难度的题目啊。。
不过 MySQL 中没有这种语法,学习一下应该如何实现?
学习一下大神的解法。
- 如果通过子查询的方法来排序求行号,扫描成本
N*N - 使用
cross join对一个行表进行关联,成本依旧很高 - 使用
select @a := 0产生只有一行的数据,扫描全表
---- MySQL ----
select score,
@a := @a + 1 as rank
from scores,
(select @a := 0) t
order by score desc;
-- 实现排序
针对相同的分数需要处理成相同的排名,所以需要添加辅助变量
@a := @a + (分数是否相同?0:1)
@a := @a + (@pre <> (@pre := score))
最终脚本,牛逼!
---- MySQL ----
select Score,
@a := @a + (@pre <> (@pre := Score)) as rank
from Scores,
(select @a := 0,
@pre := -1) t
order by Score desc; ---- 138ms
解释一下:
@a类似在Oracle中的rownum,可以在生成结果内附加上一列序列号,可以近似理解为查询后加上行号;@a := @a + 1实际上是赋值,旧值+1变为新值赋给@a;- 实际上并没有直接
@a+1那么简单,还要先去判断分数是否与前一行相同,所以引入@pre来记录:- 先将
Score赋值给:@pre := Score; - 然后判断之前的
@pre是否与赋值后的@pre不相同 ==>(@pre <> (@pre := Score))“<>” 就是 "!=" 的意思; - 两者不同,判断结果为真,则取1; 两者相同,判断结果为假,取0, 最后再用 0或1 加上
@a即为当前行分数的排名;
- 先将
(select @a := 0, @pre := -1) t为初始化@a和@pre的开始值;@pre初始值为 -1 为的是防止Score有 0 分的出现;- @第一次比较Score值的时候,
@pre初始值肯定和Score不同, 所以第一次比较@a必然会 +1,所以@a要从0开始;
- 最后以将结果根据
Score进行倒序展示; - 顺便一提的是,这个代码的结果生成的rank带有小数点,可以考虑使用
cast()方法来消除排名的小数点;
另一种,比较一般的解法。
---- MySQL ----
# Write your MySQL query statement below
select a.Score,
(select count(distinct b.Score)
from Scores b
where b.Score > a.Score) + 1 as rank
from Scores a
order by Score desc; ---- 605ms
自连接的解法速度比较慢。
思考
区分对比一下几个排序函数 row_number()、rank()、dense_rank()、ntile() 的区别。
MySQL 中可以使用变量的方式进行求解。
LeetCode:178.分数排名的更多相关文章
- SQL Server实现 LeetCode 178 分数排名
178. 分数排名 SQL架构 编写一个 SQL 查询来实现分数排名.如果两个分数相同,则两个分数排名(Rank)相同.请注意,平分后的下一个名次应该是下一个连续的整数值.换句话说,名次之间不应该有& ...
- LeetCode 178. 分数排名
1.题目描述 编写一个 SQL 查询来实现分数排名.如果两个分数相同,则两个分数排名(Rank)相同.请注意,平分后的下一个名次应该是下一个连续的整数值.换句话说,名次之间不应该有“间隔”. +--- ...
- 178. 分数排名 + MySql + RANK() OVER
178. 分数排名 LeetCode_MySql_178 题目描述 题解分析 排名函数 DENSE_RANK().如果使用 DENSE_RANK() 进行排名会得到:1,1,2,3,4. RANK() ...
- LeetCode(数据库):分数排名
,)); Truncate table Scores; ', '3.5'); ', '3.65'); ', '4.0'); ', '3.85'); ', '4.0'); ', '3.65'); 编写一 ...
- sql 178. 分数排名
编写一个 SQL 查询来实现分数排名.如果两个分数相同,则两个分数排名(Rank)相同.请注意,平分后的下一个名次应该是下一个连续的整数值.换句话说,名次之间不应该有“间隔”. +----+----- ...
- [SQL]LeetCode178. 分数排名 | Rank Scores
Write a SQL query to rank scores. If there is a tie between two scores, both should have the same ra ...
- MYSQL分数排名
编写一个 SQL 查询来实现分数排名.如果两个分数相同,则两个分数排名(Rank)相同.请注意,平分后的下一个名次应该是下一个连续的整数值.换句话说,名次之间不应该有“间隔”. +----+----- ...
- mysql查询之分数排名
编写一个 SQL 查询来实现分数排名.如果两个分数相同,则两个分数排名(Rank)相同 +----+-------+ | Id | Score | +----+-------+ | 1 | 3.50 ...
- Django项目:CRM(客户关系管理系统)--75--65PerfectCRM实现CRM课程分数排名
# classtop_urls.py # ————————64PerfectCRM实现CRM课程排名详情———————— from django.conf.urls import url from b ...
随机推荐
- Vue开发工具VS Code与调试
vscode安装 进入vscode官网(https://code.visualstudio.com/Download)vscode插件安装进入vscode官网插件商店(https://marketpl ...
- mysql密码设置为空怎么办?
由于很多童鞋安装使用MySQL时,安装时没有设置密码,或者像我一样图省事设置密码为空,想为其设置新密码: 1.点击 开始------>运行----在弹出的对话框中输入cmd 如下图: 2.使用 ...
- 19 个强大、有趣、又好玩的 Linux 命令!
民工哥技术之路 今天 点击上方“民工哥技术之路”选择“置顶或星标” 每天10点为你分享不一样的干货 1. sl 命令 你会看到一辆火车从屏幕右边开往左边…… 安装 $ sudo apt-get ins ...
- 图解 HTTP 笔记(五)——Web 服务器
该章的主要内容是讲解与 HTTP 协作的 Web 服务器 一.用单台虚拟主机实现多个域名 基于虚拟主机的功能,可以只使用一台物理机实现多个域名的网站部署. 在互联网上,域名通过 DNS 域名解析系统可 ...
- failOnMissingWebXml
使用maven创建项目时有时在pom.xml的war处出现failOnMissingWebXml的错误,根据错误提示的原因可以知道项目是web项目,打包时打成war包. 如果项目里没有web.xml, ...
- [!] The version of CocoaPods used to generate the lockfile (1.4.0.beta.1) is higher than the version of the current executable (1.3.0.beta.1). Incompatibility issues may arise.
今天在看一个开源Demo代码的时候,需要执行pod install命令,直接报错如下: 解决方法: 执行:pod update 命令更新资源库即可.
- iOS——sqlite3的使用(iOS嵌入式关系数据库)
1>添加sqlite3动态库:libsqlite3.dylib,CoreGraphics.framework,UIKit.framework,Foundation.framework 2> ...
- Flutter 异步Future与FutureBuilder实用技巧
什么是Future? Future表示在接下来的某个时间的值或错误,借助Future我们可以在Flutter实现异步操作.它类似于ES6中的Promise,提供then和catchError的链式调用 ...
- C4model实践总结
能看到这篇文章的人,我就不用废话给你介绍C4model.vscode.plantuml这些是什么以及怎么安装了. 0. 基本语法. 1. 创建常用模板.并保存到snippet. 2.利用关系REL控制 ...
- 【FFMPEG】【ARM-Linux开发】 ffmpeg 静态库使用,undefined reference错误
原文:http://blog.csdn.net/chinazjn/article/details/7954984 ffmpeg移植到dm365上,遇到undefined reference错误: GA ...