Mysql 查询实现成绩排名
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直接根据学号就可以获得成绩排名。
Mysql 查询实现成绩排名的更多相关文章
- sql 建表以及查询---复杂查询之成绩排名
废话不说,直接建表 1.表Player USE T4st -- 设置当前数据库为T4st,以便访问sysobjects IF EXISTS(SELECT * FROM sysobjects WHERE ...
- mysql查询之分数排名
编写一个 SQL 查询来实现分数排名.如果两个分数相同,则两个分数排名(Rank)相同 +----+-------+ | Id | Score | +----+-------+ | 1 | 3.50 ...
- mysql成绩排名
关于mysql成绩排名,网上大部分只是order by简单排序,忽略了成绩相同并列名次的问题. 定义了一个表score结构为:
- MySQL实现排名并查询指定用户排名功能,并列排名功能
MySQL实现排名并查询指定用户排名功能,并列排名功能 表结构: CREATE TABLE test.testsort ( id int(11) NOT NULL AUTO_INCREMENT, ui ...
- MySQL中给自定义的字段查询结果添加排名的方法
我正在用 MySQL 客户端的时候,突然想到如果可以给查询结果添加排名该多好啊,然后就找到了一个简单的解决办法. 下面是一个示例表的数据: 然后我们要根据 Roll_No 字段进行排序并给出排名,我 ...
- oracle根据成绩排名查询某个名次段的人员
先说一下表结构 名字name 分数fenshu 表名test1,以下查询的是成绩排名为第三名和第四名,这个模板让你查随意排名段的人 select name,fenshu,mc from (se ...
- Mysql 查询练习
Mysql 查询练习 ---创建班级表 create table class( cid int auto_increment primary key, caption ) )engine=innodb ...
- MySQL查询语句
来源于网络... Sutdent表的定义 字段名 字段描述 数据类型 主键 外键 非空 唯一 自增 Id 学号 INT(10) 是 否 是 是 是 Name 姓名 VARCHAR(20) 否 否 是 ...
- mysql——查询练习
Sutdent表的定义 字段名 字段描述 数据类型 主键 外键 非空 唯一 自增 Id 学号 INT(10) 是 否 是 是 是 Name 姓名 VARCHAR(20) 否 否 是 否 否 Sex 性 ...
随机推荐
- VUE 数据请求和响应(axios)
1. 概述 1.1 简介 axios是一个基于Promise(本机支持ES6 Promise实现) 的HTTP库,用于浏览器和 nodejs 的 HTTP 客户端.具有以下特征: 从浏览器中创建 XM ...
- Codeforces 280D k-Maximum Subsequence Sum [模拟费用流,线段树]
洛谷 Codeforces bzoj1,bzoj2 这可真是一道n倍经验题呢-- 思路 我首先想到了DP,然后矩阵,然后线段树,然后T飞-- 搜了题解之后发现是模拟费用流. 直接维护选k个子段时的最优 ...
- IBM X 3650 M3服务器RAID0设置
1 进入磁盘整列设置窗口 1.1 开机在提示符页面下按[F1]进入BIOS设置 1.2 依次进入子菜单[System Settings]à[Adapters and UEFI Drivers] 1.3 ...
- grep匹配某个次出现的次数
cat file | grep -c 'xxx' 统计xxx在file中出现的行数 cat file | grep -o 'xxx' 统计xxx在file中出现的次数
- C# 中使用 Excel
using System;using System.Collections.Generic;using System.Text;using System.Reflection;using System ...
- android经典源码,很不错的开源框架
高仿最美应用项目源码 项目介绍 这是仿最美应用开发的基于mvp+rxjava+retrofit的项目,很值得学 github地址: https://github.com/JJOGGER/Beaut ...
- IO 多路复用
IO 多路复用 多路复用也是要用单线程来处理客户端并发,与其他模型相比多出了select这个模块. 程序不再直接问操作系统要数据,而是先发起一个select调用,select会阻塞直到其中某个sock ...
- 【sqli-labs】Less11~Less16
学习sqli-labs的笔记,前面的笔记内容比较详细.后面的只记录关键点了. Less11: POST注入, 有回显,有错误提示 从11题起是POST注入,发现有两个输入框.用firefox的F12查 ...
- codeforce 240E 最小树形图+路径记录更新
最小树形图的路径是在不断建立新图的过程中更新的,因此需要开一个结构体cancle记录那些被更新的边,保存可能会被取消的边和边在旧图中的id 在朱刘算法最后添加了一个从后往前遍历新建边的循环,这可以理解 ...
- MySQL基于ROW格式的数据恢复
大家都知道MySQL Binlog 有三种格式,分别是Statement.Row.Mixd.Statement记录了用户执行的原始SQL,而Row则是记录了行的修改情况,在MySQL 5.6以上的版本 ...