作者 | 不剪发的Tony老师   责编 | 晋兆雨

出品 | CSDN博客

一般来说,SQL 子查询只能引用外查询中的字段,而不能使用同一层级中其他表中的字段。例如:

-- 错误示例

SELECT d.dept_name,t.avg_salary
FROM department d
JOIN (SELECT avg(e.salary) AS avg_salaryFROM employee eWHERE e.dept_id = d.dept_id) t;
SQL 错误 [42601]: ERROR: syntax error at end of input位置:183

由于 JOIN 子句中的查询语句 t 引用了左侧 department 表中的字段,因此产生了语法错误。

为了解决以上问题,我们可以使用 PostgreSQL 提供的横向子查询(LATERAL subquery)。不过在介绍 LATERAL 关键字之前,我们先来回顾一下 SELECT 和 FROM 子句的含义。例如:

SELECT dept_id, dept_name
FROM department;

简单来说,我们可以将以上查询看作一个循环处理语句。使用伪代码实现的以上 SQL 语句如下:

for dept_id, dept_name in department
loopprint dept_id, dept_name
end loop

对于 department 中的每一条记录,都执行 SELECT 语句指定的操作,以上示例简单的输出了每行记录。

SELECT 就像一个循环语句,而 LATERAL 就像是一个嵌套循环语句,对于左侧表中的每行记录执行一次子查询操作。例如,通过增加 LATERAL 关键字,我们可以修改第一个示例:

SELECT d.dept_name,t.avg_salary
FROM department d
CROSS JOIN LATERAL(SELECT avg(e.salary) AS avg_salaryFROM employee eWHERE e.dept_id = d.dept_id) t;dept_name  |avg_salary            |
-----------+----------------------+
行政管理部  |    26666.666666666667|
人力资源部  |13166.6666666666666667|
财务部      | 9000.0000000000000000|
研发部      | 7577.7777777777777778|
销售部      | 5012.5000000000000000|
保卫部      |                      |

CROSS JOIN LATERAL 右侧的查询可以引用左侧表中的字段,以上语句为 JOIN 左侧的每个部门返回了月薪总和。

LATERAL 可以帮助我们实现一些有用的分析功能,例如以下查询返回了每个部门月薪最高的 3 名员工:

SELECT d.dept_name, t.emp_name, t.salary
FROM department d
LEFT JOIN LATERAL(SELECT emp_name, salaryFROM employee eWHERE e.dept_id = d.dept_idORDER BY salary DESCLIMIT 3) t
ON TRUE;dept_name  |emp_name|salary  |
-----------+--------+--------+
行政管理部  |刘备     |30000.00|
行政管理部  |关羽     |26000.00|
行政管理部  |张飞     |24000.00|
人力资源部  |诸葛亮   |24000.00|
人力资源部  |黄忠     | 8000.00|
人力资源部  |魏延     | 7500.00|
财务部      |孙尚香   |12000.00|
财务部      |孙丫鬟   | 6000.00|
研发部      |赵云     |15000.00|
研发部      |周仓     | 8000.00|
研发部      |关兴     | 7000.00|
销售部      |法正     |10000.00|
销售部      |简雍     | 4800.00|
销售部      |孙乾     | 4700.00|
保卫部      |         |        |

对于 department 中的每个部门,子查询 t 最多返回 3 个员工信息。我们使用了 LEFT JOIN LATERAL,从而保证了“保卫部”也会返回一条数据。

同样使用伪代码表示以上查询语句:

for d in department
loopfor e in employee order by salary descloopcnt++if cnt <= 3thenreturn eelsegoto next dendend loop
end loop

通过 EXPLIAN 命令查看以上语句的执行计划:

EXPLAIN
SELECT d.dept_name, t.emp_name, t.salary
FROM department d
LEFT JOIN LATERAL(SELECT emp_name, salaryFROM employee eWHERE e.dept_id = d.dept_idORDER BY salary DESCLIMIT 3) t
ON TRUE;QUERY PLAN                                                                                       |
-------------------------------------------------------------------------------------------------+
Nested Loop Left Join  (cost=8.17..4439.35 rows=540 width=250)                                   |->  Seq Scan on department d  (cost=0.00..15.40 rows=540 width=122)                            |->  Limit  (cost=8.17..8.17 rows=1 width=132)                                                  |->  Sort  (cost=8.17..8.17 rows=1 width=132)                                             |Sort Key: e.salary DESC                                                            |->  Index Scan using idx_emp_dept on employee e  (cost=0.14..8.16 rows=1 width=132)|Index Cond: (dept_id = d.dept_id)                                            |

Nested Loop Left Join 说明 PostgreSQL 使用的就是嵌套循环算法。

版权声明:本文为CSDN博主「不剪发的Tony老师」的原创文章。

原文链接:https://blog.csdn.net/horses/article/details/118769805

☞贾跃亭回应是否有回国打算:那必须的;顺丰将拿出 2 亿元给快递员加薪;OpenAI 宣布解散机器人团队|极客头条☞“为了拿下 Offer,我在技术面试时迎合面试官,给了错误答案!”☞Windows、Linux 纷纷被爆漏洞,黑客可直取 root 权限!

什么是 PostgreSQL 横向子查询?相关推荐

  1. postgresql 子查询_PostgreSQL子事务及性能分析

    作者介绍Laurenz Albe:Cybertec的高级顾问和支持工程师.自2006年以来,一直与PostgreSQL合作并为其做出贡献. 译者简介陈雁飞:开源PostgreSQL爱好者,一直从事Po ...

  2. PostgreSQL 视图、子查询

    视图 表里面保存的是实际数据,视图里面保存的是SELECT语句(视图本身不存储数据). 从视图中读取数据,此时视图在内部执行SELECT语句,创建一张临时表. 使用视图的好处:其一,视图不保存数据,节 ...

  3. PostgreSQL 10.1 手册_部分 II. SQL 语言_第 9 章 函数和操作符_9.22. 子查询表达式

    9.22. 子查询表达式 9.22.1. EXISTS9.22.2. IN9.22.3. NOT IN9.22.4. ANY/SOME9.22.5. ALL9.22.6. 单一行比较 本节描述Post ...

  4. MySQL 学习笔记(16)— 子查询(单行单列、一行多列、多行多列、 ALL、ANY、SOME 运算符、EXISTS 操作符)

    1. 子查询概念 子查询是指嵌套在其他语句(SELECT . INSERT . UPDATE . DELETE 等)中的 SELECT 语句:子查询也称为内查询( inner query )或者嵌套查 ...

  5. SQL优化之一则MySQL中的DELETE、UPDATE 子查询的锁机制失效案例

    关注"数据和云",精彩不容错过 前言 开发与维护人员避免不了与 in/exists.not in/not exists 子查询打交道,接触过的人可能知道 in/exists.not ...

  6. oracle多表嵌套查询使用,oracle sql 多表 嵌套子查询 连接查询, join where exist i...

    转:http://hi.baidu.com/delphi_relive/blog/item/d7c0034a49c4932208f7ef21.html in 和 exists也是很好区别的. in 是 ...

  7. MySql基础篇---002 SQL之SELECT使用篇: 基本的SELECT语句,运算符,排序与分页,多表查询,单行函数,聚合函数,子查询

    第03章_基本的SELECT语句 讲师:尚硅谷-宋红康(江湖人称:康师傅) 官网:http://www.atguigu.com 1. SQL概述 1.1 SQL背景知识 1946 年,世界上第一台电脑 ...

  8. PostgreSQL中的查询:1.查询执行阶段

    PostgreSQL中的查询:1.查询执行阶段 开始关于PG内部执行机制的文章系列.这一篇侧重于查询计划和执行机制. 本系列包括: 1.查询执行阶段(本文) 2.统计数据 3.顺序扫描 4.索引扫描 ...

  9. SQL插入数据时连表查询(利用子查询一次性 insert 多条数据)

    标准说明: INSERT 语句中的 SELECT 子查询可用于将一个或多个表或视图中的值添加到另一个表中.使用 SELECT 子查询还可以同时插入多行. INSERT INTO 表名 as a (a. ...

最新文章

  1. Python3 字符串
  2. Markdown 语法介绍
  3. 怎样保证客户端和服务器端数据的一致性(数据的同步)
  4. CentOS7静默安装oracle11g
  5. 在斯坦福,做 Manning 的 phd 要有多强?
  6. Dom4j完整教程~属性相关
  7. Vue.js 服务器端渲染指南
  8. thinkphp 前台html调用函数 格式化输出
  9. 条件指示符 #ifndef 的用处
  10. halcon-高速下载
  11. Windows传统取证的一些笔记
  12. phpbb简体中文语言包的安装
  13. HP服务器raid卡型号b110i安装2008R2认不到硬盘缺少硬盘驱动
  14. easyPOI 模板导出Excel
  15. Git 常用回滚撤销命令总结
  16. 苹果计算机音频无法使用,解决Mac电脑直播没有电脑内声音的问题
  17. (42.1)【操作系统漏洞发现专题】操作系统漏洞之简介、分类、危害、发现工具、利用
  18. 前端开发日报:20190818
  19. 以太坊_Mist客户端下载及安装
  20. 常用的python读写函数

热门文章

  1. 【Linux】CentOS 7 安装 docker
  2. centos7 安装Gitlab
  3. python os
  4. ubuntu 操作记录
  5. 20个有趣的Linux命令
  6. spring中配置数据源
  7. 3 HTMLJS等前端知识系列之javascript的基础
  8. Microsoft Project学习系列(一)
  9. 【SQL数据库】查询语句练习题
  10. 在VMware中的Ubuntu虚拟机安装open-vm-tools调整屏幕可以复制粘贴