一条SQL语句面试题:求选修所有课程的学生

前几天求职面试,有一道SQL题:给出三个表:学生、课程、成绩,求选修了所有课程的学生。

一道看似很简单的问题,把我难住了,我改了又改,涂涂画画,抓耳挠腮,因为试卷没有多少空白位置了,最后只好放弃。心情大受影响,尽管最后还是获得offer。

但是心中有愧呀!

于是在机器上试了试:

先建好表

use test;
go

create table student(sno varchar(50) not null,name varchar(50) not null);
insert into student(sno,name) values(‘001‘,‘张三‘);
insert into student(sno,name) values(‘002‘,‘李四‘);
insert into student(sno,name) values(‘003‘,‘王五‘);

create table class(cno varchar(50) not null,name varchar(50) not null)
insert into class(cno,name) values(‘c01‘,‘数据结构‘);
insert into class(cno,name) values(‘c02‘,‘操作系统‘);
insert into class(cno,name) values(‘c03‘,‘计算机组成原理‘);
insert into class(cno,name) values(‘c04‘,‘网络基础‘);

create table score(sno varchar(50) not null,cno varchar(50) not null,score decimal(18,2) not null)
insert into score(sno,cno,score) values(‘001‘,‘c01‘,80);
insert into score(sno,cno,score) values(‘001‘,‘c02‘,85);
insert into score(sno,cno,score) values(‘001‘,‘c03‘,89);
insert into score(sno,cno,score) values(‘001‘,‘c04‘,87);
insert into score(sno,cno,score) values(‘002‘,‘c01‘,80);
insert into score(sno,cno,score) values(‘003‘,‘c04‘,70);

我想到了三种写法:

1、

[sql] view plaincopy

  1. select * from student s
  2. where not exists(select 1 from class c
  3. where not exists(select 1 from score
  4. where sno=s.sno and cno=c.cno));

两个not exists。我当时是只写了一个。

由内嵌到外部,

1)不存在一门为当前学生所选修的课程

select 1 from class c where not exists(select 1 from score where sno=s.sno and cno=c.cno)

2)不存在 情况1),也就是不存在当前学生有没选修的课程这种情况

换言之,当前学生选修了所有的课程

2、

[sql] view plaincopy

  1. select * from student where sno not in(
  2. select st.sno from student st,class c
  3. where not exists(select 1 from score
  4. where sno=st.sno and cno=c.cno)
  5. )

嵌套里面的语句是有未选修课程的学生

然后外部是不在此名单内的学生

3、

[sql] view plaincopy

  1. select * from student where sno in(
  2. select st.sno from student st
  3. inner join score sc on st.sno=sc.sno
  4. group by st.sno
  5. having count(*)=(select count(*) from class)
  6. )

这个语句比较容易理解。但效率可能不高,我不确定(select count(*) from class)是否要执行很多次。

时间: 2024-10-22 13:02:24

一条SQL语句面试题:求选修所有课程的学生的相关文章

关于使用一条SQL语句 找出同时符合多个tag条件的记录集合算法

表结构 Tag Table:{tag_id, tag_name}  #标签表 News Table:{news_id, title,......}  #新闻表 NewsTags Table:{tag_id, news_id}  #新闻的标签关系表 解释: 一条新闻,有多个tag标签,例如: 新闻a{Tag1,Tag2, Tag3, Tag4} 新闻b{Tag1,Tag6, Tag7, Tag8} 新闻c{Tag8,Tag9, Tag10, Tag1} 新闻...{Tag..., .....} 搜

腾讯面试:一条SQL语句执行得很慢的原因有哪些?---不看后悔系列

说实话,这个问题可以涉及到 MySQL 的很多核心知识,可以扯出一大堆,就像要考你计算机网络的知识时,问你"输入URL回车之后,究竟发生了什么"一样,看看你能说出多少了. 之前腾讯面试的实话,也问到这个问题了,不过答的很不好,之前没去想过相关原因,导致一时之间扯不出来.所以今天,我带大家来详细扯一下有哪些原因,相信你看完之后一定会有所收获,不然你打我. 一.开始装逼:分类讨论 一条 SQL 语句执行的很慢,那是每次执行都很慢呢?还是大多数情况下是正常的,偶尔出现很慢呢?所以我觉得,我们

用一条SQL语句查出每门课都大于80分的学生的姓名

用一条sql语句查询出所有课程都大于80分的学生名单: name cource score 张三 语文 81 张三 数学 75 李四 语文 76 李四 数学 90 王五 语文 81 王五 数学 100 王五 英语 90 1 SET FOREIGN_KEY_CHECKS=0; 2 3 -- ---------------------------- 4 -- Table structure for grade 5 -- ---------------------------- 6 DROP TABL

mysql(1)—— 详解一条sql语句的执行过程

SQL是一套标准,全称结构化查询语言,是用来完成和数据库之间的通信的编程语言,SQL语言是脚本语言,直接运行在数据库上.同时,SQL语句与数据在数据库上的存储方式无关,只是不同的数据库对于同一条SQL语句的底层实现不同罢了,但结果相同.这有点类似于java中接口的作用,一个接口可以有不同的实现类,不同的实现类对于接口中方法的实现方式可以不同,结果可以相同.这里SQL语言的作用就类似于java中的接口,数据库就类似于java中接口的实现类,SQL语句就类似于java接口中的方法.不同的是java中

详解一条sql语句的执行过程

SQL是一套标准,全称结构化查询语言,是用来完成和数据库之间的通信的编程语言,SQL语言是脚本语言,直接运行在数据库上.同时,SQL语句与数据在数据库上的存储方式无关,只是不同的数据库对于同一条SQL语句的底层实现不同罢了,但结果相同.这有点类似于java中接口的作用,一个接口可以有不同的实现类,不同的实现类对于接口中方法的实现方式可以不同,结果可以相同.这里SQL语言的作用就类似于java中的接口,数据库就类似于java中接口的实现类,SQL语句就类似于java接口中的方法.不同的是java中

数据库 将多条SQL语句查询的内容整合成为一个元组

1.情景再现 分类信息 住户 租户 快递 亲友访问 花苑 138 25 44 89 新城 34 150 78 34 以上的一张表中,假如要分别在两个住宅小区统计各类进出的人数,进出的类型可以分为是住户,租户,快递以及亲友访问四个类型. 但是实际的表设计中,住户以及租户是属于这个小区,可以直接统计出来,但是快递以及亲友访问则需要从进出记录查找. 换言之,就是表中的每个元组的数据是从多个表中查出来的,同时可能是用四条sql语句中查询出来的.使用如下表达: ①: select t.name as co

MyBatis插件及示例----打印每条SQL语句及其执行时间

Plugins 摘一段来自MyBatis官方文档的文字. MyBatis允许你在某一点拦截已映射语句执行的调用.默认情况下,MyBatis允许使用插件来拦截方法调用 Executor(update.query.flushStatements.commint.rollback.getTransaction.close.isClosed) ParameterHandler(getParameterObject.setParameters) ResultSetHandler(handleResultS

用一条SQL语句取出第 m 条到第 n 条记录的方法

原文:用一条SQL语句取出第 m 条到第 n 条记录的方法   --从Table 表中取出第 m 条到第 n 条的记录:(Not In 版本) SELECT TOP n-m+1 * FROM Table WHERE (id NOT IN (SELECT TOP m-1 id FROM Table )) --从TABLE表中取出第m到n条记录 (Exists版本) SELECT TOP n-m+1 * FROM TABLE AS a WHERE Not Exists (Select * From 

一条sql语句具体执行过程

客户端发送一条查询语句给服务器 服务器先检查缓存,如果有就直接返回数据,否则进入下一个阶段. 服务器进行sql解析,预处理,再由优化器生成对应的执行计划 根据执行计划,调用API接口来执行查询 返回结果给客户端,并缓存 一条sql语句具体执行过程,布布扣,bubuko.com