SQL - 子查询

  • 子查询可以写的位置:select、from、where

  • 子查询可以返回一个值:一个列的一些值,一个结果集(表)

  • 子查询的作用:可以作为条件判断的范围,作为判断条件,可以返回特定结果值,

  • 与子查询相关:

    • in 运算符;

      • in (子查询),其中子查询返回的结果是一个结果集
      sql 复制代码
      select *
      from clients
      where client_id in (
      select client_id
      from invoices
      group by client_id
      having count(*)>2)
    • 子查询 vs 连接;

      • 子查询和连接,都能用来查询信息,比较的是可读性和查询效率
      sql 复制代码
      select *
      from clients
      where client_id not in (
      select  distinct client_id 
      from invoices)
      -- 同一个问题,连接和子查询的比较
      select *
      from clients 
      left join invoices using (client_id)
      where invoice_id is null
    • all 关键字

      • 表示一个值要比all修饰的子查询返回的一列值都大,或都小等;同理可得,any关键字和some关键字都是类似的含义。
      sql 复制代码
      select *
      from invoices
      where invoice_total > all(
      select invoice_total
      from invoices
      where client_id=3)
      -- 等价于
      -- select *
      -- from invoices
      -- where invoice_total > (
      -- select max(invoice_total)
      -- from invoices
      -- where client_id=3)
    • 相关子查询

      • 子查询和外查询存在相关性,子查询的返回结果集和外查询存在联系,但是相关子查询经常执行得很慢
      sql 复制代码
      select *
      from employees e
      where salary >(
      select avg(salary)
      from employees 
      where office_id=e.office_id)
      -- 也可以写这样,但是体现不了相关子查询的特点
      -- select *
      -- from employees
      -- join (select office_id,avg(salary) as aaa
      -- from employees
      -- group by office_id) o
      -- on o.office_id=employees.office_id
      -- where salary>aaa
    • exists 运算符(存在)

      • where exists (子查询)
      • 能够提高效率,在子查询的过程中,找到一行符合条件的记录就返回true
      sql 复制代码
      -- exists
      select *
      from clients c
      where exists (
      select client_id
      from clients
      where client_id=c.client_id)
    • select 子句中的子查询
    sql 复制代码
    select 
    	invoice_id,
        invoice_total,
        (select avg(invoice_total) from invoices) as invoice_avg,
        invoice_total- (select invoice_avg) as defference
    from invoices;
     
    select client_id,name,sum(invoice_total),
    (select avg(invoice_total) from invoices) as average,
    sum(invoice_total)-(select average) as difference
    from clients
    left join invoices using (client_id)
    group by client_id,name
    • from 子句中的子查询

      • 子查询的结果可以作为一张真实的表使用
      sql 复制代码
      select *
      from employees
      join (select office_id,avg(salary) as aaa
      from employees
      group by office_id) o
      on o.office_id=employees.office_id
      where salary>aaa
相关推荐
GM_82842 分钟前
【Go项目基建】GORM框架实现SQL校验拦截器(完整源码+详解)
sql·golang·拦截器·gorm·慢查询·持久层基建
MAGICIAN...6 小时前
【Redis】--持久化机制
数据库·redis·缓存
我真的是大笨蛋6 小时前
JVM调优总结
java·jvm·数据库·redis·缓存·性能优化·系统架构
步步为营DotNet8 小时前
5-2EFCore性能优化
数据库·性能优化·.net
2501_920047038 小时前
Redis-集群
数据库·redis·bootstrap
半夏陌离9 小时前
SQL 拓展指南:不同数据库差异对比(MySQL/Oracle/SQL Server 基础区别)
大数据·数据库·sql·mysql·oracle·数据库架构
旋转的油纸伞9 小时前
SQL表一共有几种写入方式
数据库·sql
半夏陌离9 小时前
SQL 入门指南:排序与分页查询(ORDER BY 多字段排序、LIMIT 分页实战)
java·前端·数据库
isyoungboy9 小时前
SQL高效处理海量GPS轨迹数据:人员gps轨迹数据抽稀实战指南
数据库·sql
敬业小码哥10 小时前
记一次:mysql的json及json数组使用组合使用
数据库·mysql·json