Mysql 查询实现成绩排名,相同分数名次相同,类似于rank()函数

近日系统要实现总分成绩排名,而且相同分数的学生排名要一样,在网上搜了一圈,没有找到合适的方法,只能靠自己实现了,这里提供两种方法

//还有其他排名方式可以借鉴https://blog.csdn.net/a9925/article/details/76804951

1、sql查询实现

测试如下:

mysql> select * from score ;
+----------+--------------+---------------------+--------------+-------+
| study_no | student_name | subject_id | subject_name | score |
+----------+--------------+---------------------+--------------+-------+
| student1 | student1 | CodeCourseSubject_0 | 语文 | 120 |
| student2 | student2 | CodeCourseSubject_0 | 语文 | 110 |
| student3 | student3 | CodeCourseSubject_0 | 语文 | 110 |
| student4 | student4 | CodeCourseSubject_0 | 语文 | 80 |
| student5 | student5 | CodeCourseSubject_0 | 语文 | 81 |
| student1 | student1 | CodeCourseSubject_2 | 英语 | 150 |
| student2 | student2 | CodeCourseSubject_2 | 英语 | 130 |
| student3 | student3 | CodeCourseSubject_2 | 英语 | 130 |
| student4 | student4 | CodeCourseSubject_2 | 英语 | 44 |
| student5 | student5 | CodeCourseSubject_2 | 英语 | 45 |
+----------+--------------+---------------------+--------------+-------+
10 rows in set

首先这里对科目显示进行了行列转换,并且对以学号study_no进行分组计算总分,并且按排名排序, 
sql如下:

SELECT @rownum:=@rownum+1 AS rownum,
   if(@total=total,@rank,@rank:=@rownum)as rank,
   @total:=total,
   A.*
   FROM (SELECT study_no AS studyNo,
        student_name AS studentName,
        SUM(score) AS total,
        SUM(IF(subject_id='CodeCourseSubject_0',score,0)) AS 语文,
        SUM(IF(subject_id='CodeCourseSubject_2',score,0)) AS 英语
        FROM score GROUP BY study_no ORDER BY total DESC
       )A,(SELECT @rank:=0,@rownum:=0,@total:=null)B

  

结果:

+--------+------+---------------+----------+-------------+-------+-------+-------+
| rownum | rank | @total:=total | studyNo | studentName | total | 语文 | 英语 |
+--------+------+---------------+----------+-------------+-------+-------+-------+
| 1 | 1 | 270.0 | student1 | student1 | 270.0 | 120.0 | 150.0 |
| 2 | 2 | 240.0 | student2 | student2 | 240.0 | 110.0 | 130.0 |
| 3 | 2 | 240.0 | student3 | student3 | 240.0 | 110.0 | 130.0 |
| 4 | 4 | 126.0 | student5 | student5 | 126.0 | 81.0 | 45.0 |
| 5 | 5 | 124.0 | student4 | student4 | 124.0 | 80.0 | 44.0 |
+--------+------+---------------+----------+-------------+-------+-------+-------+
5 rows in set

可见排名第二名有两个相同的分数,后面的排名自动往后移。

下面提供另外一种在程序中实现的方法:

2、程序实现

首先获得全班总分并且以降序排序

public List<Object[]> findAllTotalScore() {
String sql = "SELECT study_no,sum(score) AS total from score s ";
javax.persistence.Query query = em.createNativeQuery(sql);
return query.getResultList();
}

对获得的总分信息进行排序和封装成map(study_no,rank)

private Map<String, Long> rank(List<Object[]> objects) {
long previousRank = 1;
double total = 0;
//记录排名的map,map<study_no,排名>
Map<String, Long> rankMap = new HashMap<>();
for (int i = 0; i < objects.size(); i++) {
Object[] object = objects.get(i);
//计算名次,相同分数排名一样
if (total == (double) object[1]) {
rankMap.put(object[0].toString(), previousRank);
} else {
rankMap.put(object[0].toString(), i + 1L);
total = (double) object[1];
previousRank = i + 1;
}
}
return rankMap;
}

使用map直接根据学号就可以获得成绩排名。

转自:https://blog.csdn.net/a56508820/article/details/49663069

Mysql 查询实现成绩排名的更多相关文章

  1. sql 建表以及查询---复杂查询之成绩排名

    废话不说,直接建表 1.表Player USE T4st -- 设置当前数据库为T4st,以便访问sysobjects IF EXISTS(SELECT * FROM sysobjects WHERE ...

  2. mysql查询之分数排名

    编写一个 SQL 查询来实现分数排名.如果两个分数相同,则两个分数排名(Rank)相同 +----+-------+ | Id | Score | +----+-------+ | 1 | 3.50 ...

  3. mysql成绩排名

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

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

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

  5. MySQL中给自定义的字段查询结果添加排名的方法

    我正在用 MySQL 客户端的时候,突然想到如果可以给查询结果添加排名该多好啊,然后就找到了一个简单的解决办法. 下面是一个示例表的数据:  然后我们要根据 Roll_No 字段进行排序并给出排名,我 ...

  6. oracle根据成绩排名查询某个名次段的人员

    先说一下表结构  名字name  分数fenshu   表名test1,以下查询的是成绩排名为第三名和第四名,这个模板让你查随意排名段的人 select name,fenshu,mc from (se ...

  7. Mysql 查询练习

    Mysql 查询练习 ---创建班级表 create table class( cid int auto_increment primary key, caption ) )engine=innodb ...

  8. MySQL查询语句

    来源于网络... Sutdent表的定义 字段名 字段描述 数据类型 主键 外键 非空 唯一 自增 Id 学号 INT(10) 是 否 是 是 是 Name 姓名 VARCHAR(20) 否 否 是 ...

  9. mysql——查询练习

    Sutdent表的定义 字段名 字段描述 数据类型 主键 外键 非空 唯一 自增 Id 学号 INT(10) 是 否 是 是 是 Name 姓名 VARCHAR(20) 否 否 是 否 否 Sex 性 ...

随机推荐

  1. sudo初级授权设置

    linux中,不可能人人都是用root用户去修改一些文件或者操作,所以一般需要用到对用户的权限控制,linux中可以是sudo来实现 首先,权限控制的文件中 /etc/sudoers中进行配置,文件为 ...

  2. 高性能MySQL(第3版) 中文PDF带目录清晰版

    下载地址: <高性能MySQL(第3版)>编辑推荐:"只要你不敢以MySQL专家自诩,又岂敢错过这本神书?""一言以蔽之,写得好,编排得好,需要参考时容易到爆 ...

  3. Confluence 6 配置服务器基础地址

    服务器基础地址(Server Base URL)是用户访问 Confluence 的 URL 地址.这个基础的 URL 地址必须与你在浏览器中访问 Confluence 中的地址. Confluenc ...

  4. Allegro PCB Design GXL (legacy) 设置十字大光标

    Allegro PCB Design GXL (legacy) version 16.6-2015 1.菜单:Setup > User Preferences... 2.User Prefere ...

  5. wampserver本地配置域名映射

    本地开发时,一般是在浏览器输入 http://localhost/项目文件夹名 来测试网页文件,你有没有想过在本地在浏览器输入你自己设定的一个域名进入项目文件夹中去,本地配置多域名可以测试二级域名以及 ...

  6. java反射机制实例

    1.通过数据库字段反射给对象实例赋值 convertClass.java /** * 通过反射构造对象 * @param obj * @param rs * @return */ public Obj ...

  7. Centos7上搭建ftp服务器

    ftp服务器搭建 1.安装好centos系统,配好yum仓库 其中vsftpd源在这下载 http://rpmfind.net/linux/rpm2html/search.php?query=vsft ...

  8. 饮冰三年-人工智能-Python-16Python基础之迭代器、生成器、装饰器

    一:迭代器: 最大的特点:节省内存 1.1 迭代器协议 a:对象必须提供一个next方法, b:执行方法要么返回迭代中的下一项,要么抛弃一个Stopiteration异常, c:只能向后不能向前. 1 ...

  9. 设置IDEA中的web

  10. 如何让微信里的html应用弹出“点击右上角分享到朋友圈”的图片

    一个分享按钮,一个隐藏的图片(这个图片绝对定位在右上角)然后就是点击显示,点击隐藏了... <a href="javascript:;" onclick="docu ...