这道经典SQL面试问题你会吗?

2023-11-03 20:20

本文主要是介绍这道经典SQL面试问题你会吗?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

大家经常自嘲后端开发就是crud boy嘛,今天给大家看一道SQL题,我相信很多人写不出来。我们来看一下这个题目。

create table course (id int primary key,name varchar(32) not null
);
create table student (id int primary key,name varchar(32) not null
);
create table score (id int primary key,course_id int not null,student_id int not null,score int not null
);

这三张表也是我经常在面试里面有问到的,这三张表的含义是非常好理解的,但是基于这三张表可以研发出来很多复杂的业务场景SQL面试题。我们先初始化一些课程信息和学生分数信息。

insert into course values(1,`语文`),(2,`数学`),(3,`外语`);
insert into student values(1,`小张`),(2,`小王`),(3,`小马`);
insert into score values(1,1,1,80),(2,2,1,90),(3,3,1,70);
insert into score values(4,1,2,70),(5,2,2,90),(6,3,2,80);
insert into score values(7,1,3,80),(8,2,3,60),(9,3,3,70);

总分最高的学生

求总分最高的学生他的分数是多少?这个题目其实是有一些歧义的,是总分最高呢,还是单科最高呢?作为面试的你对于题意本身理解不清晰的情况下,你应该反问面试官,而不是直接就开始写。
我们把这两种情况都来写一下,先求总分最高的学生,找到这个score表,那既然是总分,我们肯定要用sum函数,那既然用到sum,那是不是要做一个分组,基于什么分组呢?目标是要求总分最高的学生,自然想到应该基于student_id去分组,通过score表按照student_id分组我们就能拿到总分及相应的student_id,但怎么查到学生的名字呢?
很多同学很自然而然就想到去join这个student的表,但是这里隐藏一个问题,就是总分最高的可能不止一个学生,我们是通过总分进行的分组,分组后只能取到其中某一个student_id。那应该怎么去做呢?其实还要再去查一遍这个score表,把相应取得这个总分的学生的所有id都查出来,然后再去跟student的做一个连接,这样才能查出完整的数据。这个思路清晰了之后,我们的SQL语句步骤如下:

  1. 查出最高分
select sum(score) from score group by student_id order by sum(score) desc limit 1
  1. 查出最高分所有的学生ID。
select distinct(student_id) from score 
group by student_id having sum(score)=(select sum(score) from score 
group by student_id order by sum(score) desc limit 1) 
  1. join学生信息表得到获得最高分的学生信息
select s.name,t.score from student s 
right join (select distinct(student_id), sum(score) from score 
group by student_id having sum(score)  = (select sum(score) from score 
group by student_id order by sum(score) desc limit 1)) t on s.id = t.student_id

单科最高的学生

大家想一下,单科最高的话应该是要通过课程去分组,同样的获取每个单科分数最高的学生可能也不止一个,所以分组后你还要去跟score再去做一次连接,基于course id跟max score再去连一下score表从而得到每一科最高分的学生的id。
所以这个问题也是分三步,第一步就是通过course id去分组查到单科的最高分,第二步就是通过这个score表再去跟这个最高分和对应的课程id去进行right join。这里为什么是right join 大家可以思考一下,这样就查到了取得这个课程最高分的所有学生,这一步写出来了,其实最后就很简单了,去连接两个学生表跟课程表就能拿到学生的名字跟课程的名字,得到最终的SQL语句:

select s.name,c.name,s2.max_score from score s1 
right join (select max(score) max_score,course_id from score 
group by course_id) on s1.course_id = s2.course_id and s1.score=s2.max_score
left join course c on c.id = s1.course_id left join student s on s.id = student_id

虽然大部分互联网公司这种自动化的SQL生成器,也并不建议在系统里面去做连接查询,但是也有很多情况,比如说产品、运营要拉一些数据,排查一些线上问题,确实要去写一些比较复杂的CQL,而且你能写出这个SQL,代表你有两个能力,第一个就是你对业务本身理解是比较到位的,第二个就是你Sql本身的功底也是比较扎实的。
互联网行业不景气,但切莫心浮气躁,打牢基本功,才能厚积薄发!

关注公众号【小白技术圈】,回复f04即可获得2023最新全套面试资料

这篇关于这道经典SQL面试问题你会吗?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/340644

相关文章

将sqlserver数据迁移到mysql的详细步骤记录

《将sqlserver数据迁移到mysql的详细步骤记录》:本文主要介绍将SQLServer数据迁移到MySQL的步骤,包括导出数据、转换数据格式和导入数据,通过示例和工具说明,帮助大家顺利完成... 目录前言一、导出SQL Server 数据二、转换数据格式为mysql兼容格式三、导入数据到MySQL数据

MySQL分表自动化创建的实现方案

《MySQL分表自动化创建的实现方案》在数据库应用场景中,随着数据量的不断增长,单表存储数据可能会面临性能瓶颈,例如查询、插入、更新等操作的效率会逐渐降低,分表是一种有效的优化策略,它将数据分散存储在... 目录一、项目目的二、实现过程(一)mysql 事件调度器结合存储过程方式1. 开启事件调度器2. 创

SQL Server使用SELECT INTO实现表备份的代码示例

《SQLServer使用SELECTINTO实现表备份的代码示例》在数据库管理过程中,有时我们需要对表进行备份,以防数据丢失或修改错误,在SQLServer中,可以使用SELECTINT... 在数据库管理过程中,有时我们需要对表进行备份,以防数据丢失或修改错误。在 SQL Server 中,可以使用 SE

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常

mysql外键创建不成功/失效如何处理

《mysql外键创建不成功/失效如何处理》文章介绍了在MySQL5.5.40版本中,创建带有外键约束的`stu`和`grade`表时遇到的问题,发现`grade`表的`id`字段没有随着`studen... 当前mysql版本:SELECT VERSION();结果为:5.5.40。在复习mysql外键约

Vue项目中Element UI组件未注册的问题原因及解决方法

《Vue项目中ElementUI组件未注册的问题原因及解决方法》在Vue项目中使用ElementUI组件库时,开发者可能会遇到一些常见问题,例如组件未正确注册导致的警告或错误,本文将详细探讨这些问题... 目录引言一、问题背景1.1 错误信息分析1.2 问题原因二、解决方法2.1 全局引入 Element

SQL注入漏洞扫描之sqlmap详解

《SQL注入漏洞扫描之sqlmap详解》SQLMap是一款自动执行SQL注入的审计工具,支持多种SQL注入技术,包括布尔型盲注、时间型盲注、报错型注入、联合查询注入和堆叠查询注入... 目录what支持类型how---less-1为例1.检测网站是否存在sql注入漏洞的注入点2.列举可用数据库3.列举数据库

Mysql虚拟列的使用场景

《Mysql虚拟列的使用场景》MySQL虚拟列是一种在查询时动态生成的特殊列,它不占用存储空间,可以提高查询效率和数据处理便利性,本文给大家介绍Mysql虚拟列的相关知识,感兴趣的朋友一起看看吧... 目录1. 介绍mysql虚拟列1.1 定义和作用1.2 虚拟列与普通列的区别2. MySQL虚拟列的类型2

关于@MapperScan和@ComponentScan的使用问题

《关于@MapperScan和@ComponentScan的使用问题》文章介绍了在使用`@MapperScan`和`@ComponentScan`时可能会遇到的包扫描冲突问题,并提供了解决方法,同时,... 目录@MapperScan和@ComponentScan的使用问题报错如下原因解决办法课外拓展总结@

mysql数据库分区的使用

《mysql数据库分区的使用》MySQL分区技术通过将大表分割成多个较小片段,提高查询性能、管理效率和数据存储效率,本文就来介绍一下mysql数据库分区的使用,感兴趣的可以了解一下... 目录【一】分区的基本概念【1】物理存储与逻辑分割【2】查询性能提升【3】数据管理与维护【4】扩展性与并行处理【二】分区的