2023瑶池大赛

本文涉及的产品
云数据库 Tair(兼容Redis),内存型 2GB
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
简介: 瑶池SQL大赛

赛题一:找出各项考试中的佼佼者

解题思路

1、分数去重

一个学生可以参加任意考试,不限次数。

若同一个考生有多条考试记录,则取最高分。

根据上述题意,在分数 testattempt 表中按照考试类型获取每个学生的最高分数,以此来保证每个学生在每门考试中只存在一个最高分。由此,我们得到第一个子表 t 。

selectMAX(`score`) asscore,
`studentId`,
`testId`from`testattempt`GROUPBY`studentId`,
`testId`


2、进行分类排序

找出每个考试中 得分最高的的考生 。

如果存在并列,将并列的考生都列举出来,直到列举的考生达到或超过三人。

  • 先对第一步表中的数据按照考试类型将分数从高到低排序,因此使用 order by t.testid,t.score desc
  • 根据同考试类型对分数进行排名,也就是相同的分数也排名相同。这里有两种方法,一种是我们可借助变量赋值的方式进行排名处理,另一种是借助于窗口函数 DENSE_RANK() 来处理。(代码参考后续截图代码)

3、根据排名取出前三

基于第二步的排名通过判断排名小于等于3来筛选出前三的学生,为方便显示,可再进行一次分数排序

方法一

SELECTtest.`name`astest,
student.`name`asstudent,
rankresult.scoreasscoreFROM  (
SELECTt.*,
-- 排名代码,如果考试 testid 与上一个不同,那么排名从1开始
@rank :=IF(
@prev_test=`testid`,
-- 如果分数 score 与上一个相同则沿用上次的排名,否则排名+1CASEWHEN@score=t.scoreTHEN@rankWHEN@score :=t.scoreTHEN@rank :=@rank+1END,
1      ) AS`rank`,
@score :=t.score,
@prev_test :=`testid`FROM      (
selectMAX(`score`) asscore,
`studentId`,
`testId`from`testattempt`GROUPBY`studentId`,
`testId`      ) astORDERBYt.`testid`,
t.`score`DESC  ) asrankresultJOIN`test`onrankresult.testid=test.`id`join`student`onrankresult.studentid=`student`.`id`whererankresult.rank<=3orderbyrankresult.testid,
rankresult.scoredesc


方法二

SELECTtest.`name`astest,
student.`name`asstudent,
rankresult.scoreasscoreFROM  (
SELECTt.*,
-- 此处使用 dense_rank() 进行不跳号排序
      (DENSE_RANK() over ( PARTITIONby`testId`ORDERBYt.scoreDESC )) ASmyrankFROM      (
selectMAX(`score`) asscore,
`studentId`,
`testId`from`testattempt`GROUPBY`studentId`,
`testId`      ) ast  ) asrankresultJOIN`test`onrankresult.testid=test.`id`join`student`onrankresult.studentid=`student`.`id`whererankresult.myrank<=3orderbyrankresult.testid,
rankresult.scoredesc


赛题二:游戏游玩情况

解题思路

1、计算总用户数

因用户可在不同日期登录,所以在数据表中存在多条记录因此需要按照 player_id 做去重后在计算用户数

SELECTCOUNT(DISTINCT (`player_id`)) asuser_countfrom`activity`


2、获取用户首次登录后7天的日期范围

按照用户 player_id 做分组,取最小的登录日期,使用 date_add 计算该日期7天内的日期,或者通过使用datediff结合变量赋值计算出距离第一次登陆时间时间差,从而得到中间表(方法一的 t2 ,方法二的 t1)。

PS:因为本题中给定的时间是日期格式,所以如果需求不算登录当天的话,date_add 的数值应该为 6。比如:5月24日的7天内的截止日期为 5月30日,而不是 5月31日。本题结果使用7做计算。

3、获取首次登录7天后登录的用户数量

使用 activity 表与步骤2中的 t2 做 join 查询,设置activity 表中的登录时间在 t2 的区间范围内,利用 group by player_id having count(1) >1,由此得到满足条件的用户。再对结果集进行 count 即可得到首次登录后7天后的登录用户数量

4、计算百分比并四舍五入保留两位小数

使用 round() 函数对步骤3中的数据除以步骤1中的数据,得到四舍五入的结果。

方法一

SELECTround(
COUNT(t3.player_id) / (
SELECTCOUNT(DISTINCT (`player_id`)) asuser_countfrom`activity`    ),
2  ) asfractionfrom  (
SELECTt1.`player_id`,
COUNT(1) asnumFROM`activity`t1join (
SELECTmin(event_date) asstart_date,
date_add(min(event_date), interval7day)asend_date,
`player_id`FROM`activity`GROUPBY`player_id`      ) t2ont1.`player_id`=t2.player_idwheret1.`event_date`BETWEENt2.start_dateandt2.end_dateGROUPBYt1.`player_id`HAVINGCOUNT(1) >1  ) t3


方法二

selectround(
t2.player_count/ (
SELECTCOUNT(DISTINCT (`player_id`)) asuser_countfrom`activity`    ),
2  ) asfractionfrom  (
SELECtcount(DISTINCT(`player_id`)) asplayer_countfrom      (
SELECT`player_id`,
`event_date`,
casewhen@player_id=`player_id`thendatediff(`event_date`, @prev_date)
when@player_id :=`player_id`then0endasdays,
casewhen@player_id1=`player_id`then@prev_datewhen@player_id1 :=`player_id`then@prev_date :=`event_date`endasfirst_login_dateFROM`activity`orderby`player_id`,
`event_date`asc      ) t1wheret1.days<=7andt1.days>0  ) t2


赛题三:计算三角形面积

解题思路

1、确认三角形计算面积公式

此处使用向量叉积的方式计算三角形面积。假设三个点坐标如下 (x1,y1,z1),(x2,y2,z2),(x3,y3,z3),那么该三角形组成的面积S的公式为:

2、获取三个点的具体坐标值

使用 join 方法分别将 triangle 表中三个点的具体坐标值,然后再使用公式计算即可。这里我们会需要用到求平方的函数 pow,以及求平方根的函数 sqrt。

结果

SELECTt.idasTriangleId,
round(
sqrt(
pow((y1-y2), 2) *pow((z1-z3), 2) +pow((z1-z2), 2) *pow((x1-x3), 2) +pow((x1-x2), 2) *pow((y1-y3), 2)
    ) *0.5,
2  ) asAreaFROM  (
SELECTt1.id,
t2.`x`asx1,
t2.`y`asy1,
t2.`z`asz1,
t3.`x`asx2,
t3.`y`asy2,
t3.`z`asz2,
t4.`x`asx3,
t4.`y`asy3,
t4.`z`asz3FROM`triangle`t1join`point`t2ont1.`pointId1`=t2.`id`join`point`t3ont1.`pointId2`=t3.`id`join`point`t4ont1.`pointId3`=t4.`id`  ) ast


相关文章
|
24天前
|
弹性计算 人工智能 自然语言处理
云工开物:阿里云弹性计算走进高校第2期,与北京大学研一学生共探AI时代下的应用创新
阿里云高校合作、弹性计算团队​于北京大学,开展了第2届​【弹性计算进校园】​交流活动。
|
3月前
|
Cloud Native 关系型数据库 分布式数据库
|
8月前
|
人工智能 关系型数据库 MySQL
|
8月前
|
关系型数据库 分布式数据库 数据库
精彩回顾 | 《国产数据库共话未来趋势》技术沙龙上海站成功举办!
2024年1月6日(周六),由阿里云 PolarDB开源社区发起,PostgreSQL中文社区、拓数派联合主办的“国产数据库共话未来趋势”技术沙龙在上海成功举行。
|
8月前
|
存储 关系型数据库 数据库
精彩回顾 | 《国产数据库共话未来趋势》技术沙龙成功举办!
12月16日周六下午,由阿里云PolarDB开源社区、玖章算术NineData、PostgreSQL中文社区、共同举办的《国产数据库共话未来趋势》技术沙龙成功举办。
|
存储 人工智能 程序员
云栖大会天池专场:见证“数据洞察创新挑战赛”荣耀加冕
天池将于11月2日上午,与各位开发者相聚云栖大会·算力馆云栖SHOW场。
1062 0
云栖大会天池专场:见证“数据洞察创新挑战赛”荣耀加冕
|
存储 缓存 架构师
阿里云数据库专家于巍荣获PostgreSQL中国技术大会“最具价值专家 MVP”奖
2023年3月3日,在由PostgreSQL中文社区主办的“第十二届PostgreSQL中国技术大会”上,阿里云数据库开源首席架构师于巍荣获“中国 PostgreSQL 最具价值专家 MVP”奖项。
阿里云数据库专家于巍荣获PostgreSQL中国技术大会“最具价值专家 MVP”奖
|
存储 人工智能 Cloud Native
2022云栖精选—2022云栖大会云原生数据库峰会 开场致辞
致辞嘉宾:李飞飞,阿里巴巴集团副总裁,阿里云数据库产品事业部负责人,ACM和IEEE会士(FELLOW)
2022云栖精选—2022云栖大会云原生数据库峰会 开场致辞
|
SQL 安全 数据管理
NineData 荣获 IT168 & ITPub "年度创新产品"奖
在IT168 & ITPUB的年度评比中,玖章算术的NineData产品脱颖而出,获得了"年度创新产品"奖。在过去的几年,获此奖项的产品有阿里云PolarDB、TiDB、Oracle自治数据库、AnalyticDB 3.0 、腾讯云TDSQL等。该奖项反应了,在过去的一年数据库领域中,新出现的极具潜力的数据库产品。多云数据库时代已经到来,NineData致力于解决多云数据管理、复制、备份等挑战,帮助开发者用好数据和云。
389 0
NineData 荣获 IT168 & ITPub "年度创新产品"奖
|
机器学习/深度学习 存储 数据采集
数脉科技亮相云栖大会 荣膺“阿里云突出贡献奖”
近日,以“计算·进化·未来”为主题的2022云栖大会在杭州云栖小镇隆重举办,大会汇聚众多重磅行业领军人物和专家学者,阿里云产品伙伴发展论坛也在大会现场举行。
255 0
数脉科技亮相云栖大会 荣膺“阿里云突出贡献奖”