选读SQL经典实例笔记16_逻辑否定

这篇具有很好参考价值的文章主要介绍了选读SQL经典实例笔记16_逻辑否定。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

选读SQL经典实例笔记16_逻辑否定文章来源地址https://www.toymoban.com/news/detail-622673.html

1. 示例数据

1.1. student

insert into student values (1,'AARON',20)
insert into student values (2,'CHUCK',21)
insert into student values (3,'DOUG',20)
insert into student values (4,'MAGGIE',19)
insert into student values (5,'STEVE',22)
insert into student values (6,'JING',18)
insert into student values (7,'BRIAN',21)
insert into student values (8,'KAY',20)
insert into student values (9,'GILLIAN',20)
insert into student values (10,'CHAD',21)

1.2. courses

insert into courses values ('CS112','PHYSICS',4)
insert into courses values ('CS113','CALCULUS',4)
insert into courses values ('CS114','HISTORY',4)

1.3. professor

insert into professor values ('CHOI','SCIENCE',400,45)
insert into professor values ('GUNN','HISTORY',300,60)
insert into professor values ('MAYER','MATH',400,55)
insert into professor values ('POMEL','SCIENCE',500,65)
insert into professor values ('FEUER','MATH',400,40)

1.4. take

insert into take values (1,'CS112')
insert into take values (1,'CS113')
insert into take values (1,'CS114')
insert into take values (2,'CS112')
insert into take values (3,'CS112')
insert into take values (3,'CS114')
insert into take values (4,'CS112')
insert into take values (4,'CS113')
insert into take values (5,'CS113')
insert into take values (6,'CS113')
insert into take values (6,'CS114')

1.5. teach

insert into teach values ('CHOI','CS112')
insert into teach values ('CHOI','CS113')
insert into teach values ('CHOI','CS114')
insert into teach values ('POMEL','CS113')
insert into teach values ('MAYER','CS112')
insert into teach values ('MAYER','CS114')

2. 问题1:没有选修过CS112课程的学生

2.1. sql

select *
  from student
 where sno in ( select sno
                  from take
                 where cno != 'CS112' )

2.2. sql

select *
  from student
 where sno not in (select sno
                     from take
                    where cno = 'CS112')

2.3. 要记住真正的逻辑否定要求两个步骤,即为了找出‘哪些人不是’,就要先找出‘哪些人是’,然后再排除掉他们

2.4. PostgreSQL

2.5. MySQL

2.6. 使用CASE表达式和聚合函数MAX标识一个学生是否选修了CS112课程

2.6.1. sql

select s.sno,s.sname,s.age
  from student s left join take t
    on (s.sno = t.sno)
 group by s.sno,s.sname,s.age
having max(case when t.cno = 'CS112'
                then 1 else 0 end) = 0

2.7. Oracle

2.7.1. group by解决方案

select s.sno,s.sname,s.age
   from student s, take t
  where s.sno = t.sno (+)
  group by s.sno,s.sname,s.age
 having max(case when t.cno = 'CS112'
                 then 1 else 0 end) = 0

2.7.2. 窗口函数解决方案

select distinct sno,sname,age
   from (
 select s.sno,s.sname,s.age,
        max(case when t.cno = 'CS112'
                 then 1 else 0 end)
        over(partition by s.sno,s.sname,s.age) as takes_CS112
   from student s, take t
  where s.sno = t.sno (+)
        ) x
  where takes_CS112 = 0

2.8. DB2

2.9. SQL Server

2.10. 使用CASE表达式和窗口函数MAX OVER

2.10.1. sql

select distinct sno,sname,age
  from (
select s.sno,s.sname,s.age,
       max(case when t.cno = 'CS112'
                then 1 else 0 end)
       over(partition by s.sno,s.sname,s.age) as takes_CS112
  from student s, take t
    on (s.sno = t.sno)
       ) x
 where takes_CS112 = 0

2.11. 外连接到TAKE表是为了确保把那些没有选修任何课程的学生也能被筛选出来

2.12. 调用MAX函数找出最大的CASE表达式返回值

3. 问题2:只选修了CS112和CS114中的一门,而不是两门都选的学生

3.1. sql

select *
  from student
 where sno in ( select sno
                  from take
                 where cno != 'CS112'
                   and cno != 'CS114' )

3.2. sql

select *
  from student s, take t
 where s.sno = t.sno
   and t.cno in ( 'CS112', 'CS114' )
   and s.sno not in ( select a.sno
                      from take a, take b
                     where a.sno = b.sno
                       and a.cno = 'CS112'
                       and b.cno = 'CS114' )

3.3. 使用自连接找出同时选修了CS112和CS114的学生

3.4. 使用子查询从选修了CS112或CS114的学生中把同时选了两门的学生剔除掉

3.5. DB2

3.6. Oracle

3.7. SQL Server

3.8. CASE表达式和窗口函数SUM OVER

3.8.1. sql

select distinct sno,sname,age
  from (
select s.sno,s.sname,s.age,
       sum(case when t.cno in ('CS112','CS114') then 1 else 0 end)
       over (partition by s.sno,s.sname,s.age) as takes_either_or
  from student s, take t
 where s.sno = t.sno
       )x
 where takes_either_or = 1

3.9. PostgreSQL

3.10. MySQL

3.11. CASE表达式和聚合函数SUM

3.11.1. sql

select s.sno,s.sname,s.age
  from student s, take t
 where s.sno = t.sno
 group by s.sno,s.sname,s.age
having sum(case when t.cno in ('CS112','CS114')
                then 1 else 0 end) = 1

3.12. 内连接STUDENT表和TAKE表,这样就排除了那些没有选修任何课程的学生

3.13. 使用CASE表达式标记一个学生是否选修了这两门课程中的一门

3.14. 函数SUM会把每个学生对应的1都累加起来

4. 问题3:选修了CS112,而且没有选修其他课程的学生

4.1. sql

select s.*
  from student s, take t
 where s.sno = t.sno
   and t.cno = 'CS112'

4.2. sql

select s.*
  from student s, take t
 where s.sno = t.sno
   and s.sno not in ( select sno
                        from take
                       where cno != 'CS112' )

4.3. 子查询负责找出至少选修了一门课,但又没有选修CS112的所有学生

4.4. 外层查询负责找出选修了一门课程(任意课程),并且不在上述子查询的返回结果的学生

4.5. STUDENT表和TAKE表之间的连接操作过滤掉没有选修任何课程的学生

4.6. PostgreSQL

4.7. MySQL

4.8. 使用聚合函数COUNT确保下列查询返回的学生只选修了一门课程

4.8.1. sql

select s.*
  from student s,
       take t1,
       (
select sno
   from take
 group by sno
having count(*) = 1
       ) t2
 where s.sno  = t1.sno
   and t1.sno = t2.sno
   and t1.cno = 'CS112'

4.8.2. 使用内嵌视图T2找出只选修了一门课程的学生

4.8.3. 连接内嵌视图T2到TAKE表,并且筛选出选修CS112课程的学生

4.8.4. 在内嵌视图T2和TAKE表连接查询的基础上再次连接STUDENT表,找出匹配的学生

4.9. DB2

4.10. Oracle

4.11. SQL Server

4.12. 使用窗口函数COUNT OVER

4.12.1. sql

select sno,sname,age
  from (
select s.sno,s.sname,s.age,t.cno,
       count(t.cno) over (
         partition by s.sno,s.sname,s.age
       ) as cnt
  from student s, take t
 where s.sno = t.sno
       ) x
 where cnt = 1
   and cno = 'CS112'

4.12.2. 窗口函数解决方案处理方式上稍有不同(更有效率)

4.12.3. 内嵌视图X返回了每一个学生、他们选修的课程以及他们选修了几门课程

4.12.4. 获得了每个学生选修的课程和课程数目之后,最后只要保留CNT等于1并且CNO等于CS112的行即可

到了这里,关于选读SQL经典实例笔记16_逻辑否定的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 选读SQL经典实例笔记09_数值处理

    8.4.1.1. Oracle Database 10g 8.4.2.1. Oracle 9i

    2024年02月17日
    浏览(69)
  • 选读SQL经典实例笔记15_窗口函数

    2.2.1.1. 分组不为空 2.2.1.1.1. 一个分组至少要拥有一个成员(行 2.2.1.1.2. 无法从一个空表中生成任何分组 2.2.1.2. 分组具有唯一性 2.2.1.2.1. 如果查询语句使用了GROUP BY子句,那么通常而言SELECT列表里就不再需要使用DISTINCT了 2.2.2.1. COUNT永远大于0 2.2.2.1.1. 无法从一个空表

    2024年02月14日
    浏览(27)
  • 选读SQL经典实例笔记08_区间查询

    1.6.3.1. 即使同一天入职的员工不止一个人,也只会返回一个值 1.6.4.1. 使用MIN函数来确保只返回一个值 2.2.2.1. PROJ_START和PROJ_END的值决定哪些行属于同一个区间 2.2.2.2. 如果某一行的PROJ_START值等于上一行的PROJ_END值,那么该行就是“连续”的,或者说它属于某个组 3.4.1.1. ora

    2024年02月16日
    浏览(87)
  • 选读SQL经典实例笔记14_层次查询

    2.6.1.1. sql 2.7.1.1. sql 3.5.1.1.  sql 4.5.1.1.  sql 5.5.1.1.  sql 6.7.1.1. Oracle Database 10g新增的CONNECT_BY_ROOT和CONNECT_BY_ISLEAF

    2024年02月15日
    浏览(95)
  • 选读SQL经典实例笔记05_日期运算(下)

    3.6.2.1. 对于MySQL 版本的DATEDIFF函数,需要省略第一个参数day,并把剩下的两个参数的顺序颠倒过来 4.3.2.1. Oracle早期版本

    2024年02月13日
    浏览(71)
  • 选读SQL经典实例笔记11_结果集变换

    3.6.1.1. 为了剔除掉Null,需要调用聚合函数MAX,并基于RN执行GROUP BY 3.9.1.1. 使用标量子查询基于EMPNO为每个员工排序 3.9.1.2. 针对标量子查询的返回值执行GROUP BY 3.9.1.3. 使用CASE表达式和聚合函数MAX实现结果集变换

    2024年02月16日
    浏览(72)
  • 选读SQL经典实例笔记04_日期运算(上)

    2.1.1.1. 因为X和Y之间没有任何连接条件,这里会产生笛卡儿积 2.1.1.2. X和Y都只有一条数据,因而即使没有连接条件也不会有问题,结果集最终只会有一行 2.8.1.1. 对于MySQL 而言,只需去掉DATEDIFF函数的第一个参数,并翻转ALLEN_HD和WARD_HD的顺序即可 3.1.2.1. sql 3.1.2.2. sql 3.1.2.2.

    2024年02月13日
    浏览(67)
  • 选读SQL经典实例笔记02_多表查询

    3.1.2.1. 排除重复项

    2024年02月12日
    浏览(131)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包