多表查询

笛卡尔积

  • 左表的每条数据和右表的每条数据组合,这种效果称为笛卡尔乘积

select * from emp, dept;

  • 笛卡尔积引入了很多无用的数据,要完成多表查询,需要设置过滤条件来消除无用的数据

连接查询

  1. 从哪些表中查询数据
  2. 关联条件是什么
  3. 查询哪些字段

内连接查询

用左边表的记录去匹配右边表的记录,如果符合条件的则显示。如:从表.外键=主表.主键

1. 隐式内连接:使用where

-- 查询所有员工信息和对应的部门信息
select * from emp ,dept where emp.DEPTNO = dept.DEPTNO; -- 查询员工表的工号\姓名\部门名称
select emp.EMPNO,emp.ENAME,dept.DEPTNO,dept.DNAME
from emp ,dept
where emp.DEPTNO = dept.DEPTNO;

2. 显式内连接:使用inner join ... on ...

select 字段列表 from 表名1 [inner] join 表名2 on 条件

-- 将隐式内连接改成显示内连接

select emp.EMPNO,emp.ENAME,dept.DEPTNO,dept.DNAME
from emp
inner join dept
on emp.DEPTNO = dept.DEPTNO; select emp.EMPNO,emp.ENAME,dept.DEPTNO,dept.DNAME
from emp
join dept
on emp.DEPTNO = dept.DEPTNO;

外连接查询

1. 左外连接

用左边表的记录去匹配右边表的记录,如果符合条件的则显示;否则,显示 NULL
:在内连接的基础上保证左表的数据全部显示(左表是部门,右表员工)
select 字段列表 from 表1 left [outer] join 表2 on 条件; -- 查询所有员工信息,如果员工有部门,则查询部门名称,没有部门,则不显示部门名称
select emp.*,dept.DNAME from emp left OUTER join dept on emp.DEPTNO = dept.DEPTNO; -- 省略OUTER关键字
select emp.*,dept.DNAME from emp left join dept on emp.DEPTNO = dept.DEPTNO; -- mysql不支持 这种写法
select emp.*,dept.DNAME from emp , dept where emp.DEPTNO = dept.DEPTNO(+);

2. 右外连接

用右边表的记录去匹配左边表的记录,如果符合条件的则显示;否则,显示 NULL
:在内连接的基础上保证右表的数据全部显示
select 字段列表 from 表1 right [outer] join 表2 on 条件; -- 右连接
select emp.*,dept.DNAME from dept RIGHT OUTER join emp on emp.DEPTNO = dept.DEPTNO; select emp.*,dept.DNAME from dept RIGHT join emp on emp.DEPTNO = dept.DEPTNO; -- mysql不支持这种写法
select emp.*,dept.DNAME from emp , dept where emp.DEPTNO(+) = dept.DEPTNO;
-- 该sql仅用于演示右外连接写法 和上面2条sql 含义不一样

3.全外连接

全外 = 左外+右外+去重

-- 全外连接 mysql不支持

select emp.empno,emp.ename,emp.deptno,dept.deptno,dept.dname
from emp
full outer join dept
on emp.deptno = dept.deptno;

自连接

将一张表,通过别名“视为”不同的表

select e.ename 员工, b.ename 领导
from emp e,emp b
where e.mgr = b.empno ;

自连接不适合操作大表 层次查询

画层次图

oracle

select level,empno,ename,mgr from emp

connect by prior empno=mgr

start with mgr is null

order by 1;

子查询

查询中嵌套查询,称嵌套查询为子查询。

示例:查询工资比SCOTT高的员工信息
第一步:查SCOTT的工资
select sal from emp where ename='SCOTT';
第二步:查询比3000高的员工
select * from emp where sal > 3000; 子查询:子查询的查询结果,作为主查询的查询条件。
以上两步骤,合二为一:
select * from emp where sal > ( select sal from emp where ename='SCOTT' );

子查询注意:

a.子查询可以出现的位置:where、select、having、from ;不能出现在group by后面。

b. 主查询和子查询可以不是同一张表

e. 一般不在子查询中排序;但在Top-N分析问题中,必须对子查询排序

子查询中的null

not in不能有null

not in (10,20,null)

示例:查询是老板的员工信息

select * from emp where empno not in (select mgr from emp);

必须排空: (select mgr from emp where mgr is not null)

子查询不同情况

1. 子查询的结果是单行单列的
可以作为条件,使用运算符去判断 运算符: > >= < <= =
也可以使用在 select 后 或 having后 -- 查询员工工资小于平均工资的人
SELECT * FROM emp WHERE emp.sal < (SELECT AVG(sal) FROM emp); 2. 子查询的结果是 *多行单列* 的
可以作为条件,使用多行运算符来判断 in
-- 查询'SALES'和'OPERATIONS'所有的员工信息
SELECT * FROM emp WHERE deptno IN (SELECT deptno FROM dept WHERE dNAME = 'SALES' OR dNAME = 'OPERATIONS'); 子查询中的运算符 any、all any: 和集合中任意一个值比较
查询工资比30号部门其中一个员工高的员工信息
select * from emp where sal > any (select sal from emp where deptno=30);
等价于 > (select min(sal) from …) all: 和集合中的所有值比较
查询工资比30号部门所有员工高的员工信息
select * from emp where sal > all (select sal from emp where deptno=30);
等价于 > (select max(sal) from…) 3. 子查询的结果是多行多列的
可以作为一张虚拟表参与查询 -- 查询员工入职日期是1986-11-11日之后的员工信息和部门信息
-- 子查询
SELECT * FROM dept t1 ,(SELECT * FROM emp WHERE emp.hiredate > '1986-11-11') t2 WHERE t1.deptno = t2.deptno; -- 注意oracle和mysql日期格式区别 oracle:'11-11月-86' -- 可以改写成普通内连接

集合

  • union:并集,返回各个查询的所有记录,重复部分,只出现一次
  • union all:并集,返回各个查询的所有记录,重复部分 不受影响
  • intersect,交集:返回两个集合 共用的记录。
  • minus,补集:返回 包含在第一个查询集合中,但同时不包含在第二个查询集合中的记录。
-- intersect 和 minus mysql不支持

select * from emp where deptno =10
union
select * from emp where deptno =20; select * from emp where deptno =10
union all
select * from emp where deptno =20; select * from emp where deptno != 10
intersect
select * from emp where deptno =20; select * from emp where deptno !=10
minus
select * from emp where deptno =20;

sql-DQL-多表联查的更多相关文章

  1. mybatis-plus注解版实现多表联查(sql)

    mybatis注解版实现多表联查 需求: 用户有角色,角色有权限,需要一次取用户信息包含角色信息及其对应权限 实体类: package cn.zytao.taosir.common.model.use ...

  2. mybatis.net 多表联查

    mybatis.net针对多表联查,其实不用讲联查出的所有的列全部做一个新的resultMap,我们完全可以通过集成关系来实现,真是上一次说的懒加载,在一定程度上可以提高其性能,但这并不是说懒加载性能 ...

  3. asp.net mvc 三层加EF两表联查

    首先打开vs软件新建项目创建web中的mvc项目再右击解决方案创建类库项目分别创建DAL层和BLL层再把DAL层和BLL层的类重命名在mvc项目中的Models文件夹创建model类在DAL创建ADO ...

  4. SQLServer多表联查,多表分页查询

    多表联查: select p.*,s.Sheng , i.Shifrom  [dbo].[ProductRecordInfo]   --表名 p left join [ShengInfo] s on ...

  5. [工作日志] 2018-11-30 重要: 1. 多条件+ 分页 + 多表联查 2. idea拉新分支

    多条件+ 分页 + 多表联查 多条件查询 1.pom依赖 <dependency> <groupId>commons-dbutils</groupId> <a ...

  6. .NET MVC+ EF+调用存储过程 多表联查以及VIEW列表显示

    直接上干活,至于网上的一大堆处理方式不予评论,做好自己的就是最好的,供大家不走弯路 1.view页面 <link href="~/Content/bootstrap.css" ...

  7. Sql 2000系统表 语句查询表结构

     SQL2000系统表的应用  –1:获取当前数据库中的所有用户表 select Name from sysobjects where xtype=’u’ and status>=0 –2:获取 ...

  8. DQL多表查询

    DQL多表查询 一.多表查询实现多个表之间查询数据 1.交叉连接笛卡尔积:A表中的每一行匹配B表中的每一行基本结构:select [数据库名1.]表名1,属性名1,......, [数据库名.]表名. ...

  9. Mybatis中多表联查,查询出来的字段出现重名,造成数据异常的解决方法!

    在做一对多出现的问题,引发的思考:当数据库表中,主表的主键id和明细表的中的字段名相同时怎么办?Mybatis进行自动映射赋值的时候会不会出现异常?                      注意:M ...

  10. 你真的会玩SQL吗?表表达式,排名函数

    你真的会玩SQL吗?系列目录 你真的会玩SQL吗?之逻辑查询处理阶段 你真的会玩SQL吗?和平大使 内连接.外连接 你真的会玩SQL吗?三范式.数据完整性 你真的会玩SQL吗?查询指定节点及其所有父节 ...

随机推荐

  1. el-menu菜单 -- unique-opened 子菜单唯一性失效

    总结: 点击的是  el-sub-menu . 所以  el-sub-menu 的唯一性是必须的.否则 unique-opened 属性不生效

  2. 网页跟随系统 dark mode (暗黑模式) 的实现

    经过几十年的沉默, dark mode(暗黑模式) 又回到了我们面前,越来越多的 APP 有了暗黑主题,越来月多的操作系统原生添加了 "全局暗黑模式", 那么一个网站如何跟随系统的 ...

  3. 在 WinForms 项目中使用全局快捷键

    借助于全局快捷键,用户可以在任何地方操控程序,触发对应的功能.但 WinForms 框架并没有提供全局快捷键的功能.想要实现全局快捷键需要跟 Windows API 打交道.本文就交你如何使用 Win ...

  4. what 的页面制作

    1. html结构 <!-- section: what we do --> <section id="what" class="bg-light py ...

  5. [AcWing 2816] 判断子序列

    点击查看代码 #include<iostream> using namespace std; const int N = 1e5 + 10; int a[N], b[N]; int mai ...

  6. FreeRTOS --(8)任务管理之创建任务

    转载自https://blog.csdn.net/zhoutaopower/article/details/107034995 在<FreeRTOS --(7)任务管理之入门篇>文章基本分 ...

  7. 使用 Ansible 快速部署 HBase 集群

    背景 出于数据安全的考虑,自研了一个低成本的时序数据存储系统,用于存储历史行情数据. 系统借鉴了 InfluxDB 的列存与压缩策略,并基于 HBase 实现了海量存储能力. 由于运维同事缺乏 Had ...

  8. TCP 协议灵魂 12 问,巩固你的网路底层基础!

    点击上方"开源Linux",选择"设为星标" 回复"学习"获取独家整理的学习资料! 先亮出这篇文章的思维导图 TCP 作为传输层的协议,是一 ...

  9. Linux系统句柄优化

    Linux系统句柄介绍 文件句柄,会随着进程数增加而增加.其实Linux是有文件句柄限制的,而且Linux默认一般都是1024.在生产环境中很容易到达这个值,因此这里就会成为系统的瓶颈. 在Linux ...

  10. 多线程07:async、future、packaged_task、promise

    async.future.packaged_task.promise 本节内容需要包含头文件:#include <future> 一.std::async. std::future 创建后 ...