求助一条 SQL 语句的写法,分组查询,求每个学生第一次考试成绩和最后一次考试成绩的差值

2020-12-29 19:04:17 +08:00
 LeeReamond

目前有一表如下,共三列

学生 ID   考试时间         成绩
0        2020-01-01      60
0        2020-02-01      70
0        2020-03-01      80
1        2020-02-14      90
1        2020-03-15      80

想要执行一条语句搜索,得到如下结果

学生 ID     第一次成绩      最后一次成绩   差值
0          60             80            20
1          90             80            -10

也就是想要通过 group by 根据学生 ID 进行分组,之后取出某组中的初次和末次。 平台 Oracle,看了一下 Oracle 分组查询的教学,似乎 Oracle 分组里面只有 max,min,avg 之类的函数 没有办法按顺序取某条某条吗?有没有大佬讲解一下,谢谢

1490 次点击
所在节点    问与答
20 条回复
kiracyan
2020-12-29 19:16:26 +08:00
order by 考试时间
loliordie
2020-12-29 19:18:31 +08:00
Order by 考试时间 和 order by 考试时间 desc 组合起来即可
uselessVisitor
2020-12-29 19:20:06 +08:00
mysql 的话我想是 按照时间 asc desc 然后 limit 1 ? 两个结果 union all 一下?。。Oracle 没用过。。
LeeReamond
2020-12-29 20:41:15 +08:00
@kiracyan
@loliordie
order by 之后呢,如何在分组中取出开头结尾?
loading
2020-12-29 20:42:59 +08:00
最好说一下是哪型数据库,有特殊语法。
loading
2020-12-29 20:43:31 +08:00
oracle 要用 row number 吧
qiayue
2020-12-29 21:15:31 +08:00
有一个特别简单的办法,用户表加两个字段,首次考试分数和最后一次考试分数,实际产品中,这是最佳解法
yeqizhang
2020-12-29 22:18:02 +08:00
楼主自己找到解法的话分享一下哈,我也学习学习
l00t
2020-12-29 23:04:28 +08:00
多年不写有些具体的忘记了,我就提几个关键字,楼主你自己去查一下补全吧。

first_value(成绩)over(partition by ID order by 时间), last_value(成绩)over(partition by ID order by 时间)
LeeReamond
2020-12-30 00:32:01 +08:00
@l00t 感谢大佬,成功实现了。不过这个我做出来的结果是,比如源数据库共有两种学生 ID,那么我期望输出两行,但是他实际输出的行数与考试次数相等,然后每一行数据都一样,这种结果出来怎么处理?

进行 group by 吗?感觉又浪费了一次算力。生产的最佳实践是什么?
dzdh
2020-12-30 00:38:57 +08:00
@LeeReamond 最佳实践看#7
l00t
2020-12-30 08:17:47 +08:00
@LeeReamond #10 加个 distinct
isnullstring
2020-12-30 08:32:46 +08:00
@qiayue 哈哈哈,没有比静态数据来更快的解法
ebony0319
2020-12-30 09:40:34 +08:00
select studentId,max(score),min(score),max(score)-min(score) from score s where 1=(
select count(1) from score s1 where s.score>=s1.score and s.studentId=s1.studentId
) or 1=(
select count(1) from score s1 where s.score<=s1.score and s.studentId=s1.studentId
) group by studentId
ebony0319
2020-12-30 09:42:05 +08:00
ebony0319
2020-12-30 09:44:21 +08:00
抱歉,审错题了.我是按照最高分,最低分来写的.吧里面的条件改成 s.time>=s1.time 即可
l00t
2020-12-30 13:18:43 +08:00
@ebony0319 #16 条件可以改,但你聚合函数打算用什么?
ebony0319
2020-12-30 13:45:42 +08:00
@l00t 不变
isir1234
2020-12-30 15:15:34 +08:00
select student_id, gap
from (select student_id,
last_value(score) over (partition by student_id order by created_at) -
first_value(score) over (partition by student_id order by created_at) AS gap,
row_number() over (partition by student_id order by created_at desc) rn
from exam) tmp
where rn = 1;
l00t
2020-12-30 17:47:00 +08:00
@ebony0319 #18 不变不就错了?第一次考试又不是最高分,第二次又不是最低分。

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/740083

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX