您的当前位置:首页>全部文章>文章详情

【MySQL】 复合查询 | 内外连接

CrazyPanda发表于:2023-11-30 21:44:10浏览:857次TAG:
【转】【MySQL】 复合查询 | 内外连接【侵删】


1. 复合查询

多表笛卡尔积

显示雇员名、雇员工资以及所在部门的名字

由于员工 信息属于 emp表 而所在部门名字属于 dept表
数据来自不同的表,所以需要进行多表查询


1.png

表示从 emp (员工表) 和dept (部门表) 中获取信息


1.png

分别用emp员工的信息 与 dept 部门表的信息 做 穷举 组合
就称 两张表 进行笛卡尔积
就把两张表 变成了 一张表 ,就可以进行单表查询


1.png

如 smith 本来是20号部门的,把10号部门的信息传过来 是没有意义的


1.png

输入 select * from emp,dept where emp.deptno =dept.deptno;
把对应的部门号 与部门信息 匹配上


1.png

输入 select ename ,sal,dname from emp,dept where emp.deptno =dept.deptno;
就可以显示雇员名、雇员工资以及所在部门的名字

自连接

自连接 指的是在同一张表连接查询


1.png

将同一张表单纯的写两次,是不可以一起查询的


1.png

可以把同一张表的表名通过重命名的方式 分别 进行不同命名 即可

把对同一张表 进行笛卡尔积的 行为 叫做 自连接


显示员工FORD的上级领导的编号和姓名

先找 员工FORD 和领导编号 ,与员工表有关
再通过领导的编号,找领导信息,也是与员工表有关
所以就需要借助 自连接


1.png

输入 select e2.ename ,e2.empno from emp e1,emp e2 where e1.ename=‘FORD’ and e1.mgr=e2.empno;
通过第一张表emp中的 员工FORD, 找到对应的领导编号,正好对应第二张emp表的员工编号
所以设置 e1.mgr =e2.empno

这样就可以 通过员工FORD 找到 对应的 领导 的名字和编号

在where子句使用子查询

子查询 是指嵌入其他sql语句中的select语句,也叫嵌套查询

单行子查询

返回一行记录的子查询

1.png

查看emp表 的员工信息


1.png

想要查看工资最高的员工的名字和工作岗位
首先 要找到 最高工资,但是发现这样写是不可以的

子查询: 在 一条sql内部 再执行 select查询
先执行内部的sql语句,再执行外部的sql语句


1.png

先找到 最高工资 ,再通过最高工资找到对应这个人的名字和工作信息


多行子查询

返回多行记录的子查询

in关键字

用来进行判断一个对应的列值是否在一个集合当中
若在集合中,则为查找成功


查询和10号部门的工作岗位相同雇员的名字、岗位、工资、部门号 ,但是不包含10号自己的

1.png

首先要找到10号部门的工作岗位


1.png

再通过in关键字,只有工作种类在 10号部门的工作岗位的集合中 ,才能返回


1.png

使用and 关键字 ,同时也不包含10号部门的人的信息


all关键字

表示所有人


显示工资比部门30的所有员工的工资高的员工的姓名 、工资 和部门号

1.png

先取到 30号部门的所有员工对应的工资
加入 distinct 去除重复的工资


1.png

all (select distinct sal from emp where deptno=30) 表示大于 30号部门的所有员工的薪资


any关键字

表示比任意一个高


显示工资比部门30的任意员工的工资高的员工的排名(包含自己部门)

1.png

先取到 30号部门的所有员工对应的工资
加入 distinct 去除重复的工资


1.png

any(select distinct sal from emp where deptno=30) 表示大于 30号部门的任意员工的薪资


多列子查询

多列子查询 是指 返回多个列数据的子查询语句


查询和SMITH的部门和岗位完全相同的所有雇员,不含SMITH本人

1.png

首先要找到SMITH对应的部门和岗位


1.png

因为要比较的是 两个条件 部门和岗位
而mysql 支持 圆括号 直接写两列
将SMITH 对应的部门和岗位作为筛选条件


1.png

还需添加条件 ename <> ‘SMITH’ 不能为SMITH员工本身


在from子句中使用子查询

在查询语句在from子句中,把一个子查询当作一个临时表使用


显示每个高于自己部门平均工资的员工的姓名、部门、工资、平均工资

1.png

首先找到每个部门的平均工资
通过group by 将各个部门分开,再获取对应的每个部门的平均工资


1.png

select deptno,avg(sal) from emp group by deptno 整体操作所形成的数据 看作一张表,并重命名为 tmp,与emp表 做 笛卡尔积


1.png

寻找到emp表的部门号 与 tmp表的部门号 相同的 数据 才是合适的,所以加上 emp.deptno=tmp.deptno


1.png

select deptno,avg(sal) myavg from emp group by deptno
为了不让其在后面比较起冲突,所以 将avg(sal) 重命名为 myavg
再添加 emp.sal > tmp.myavg 筛选条件,显示每个高于自己部门平均工资的员工信息


合并查询

为了合并多个select的执行结果,可以使用集合符 unionunion all


union

1.png

该操作符用于取得 两个结果集的并集,当使用该操作符时,会自动去掉结果集中的重复行


1.png

select * from emp where sal>2500 即工资大于2500 的人的信息


1.png

select * from emp where job=‘MANAGER’ 即工作岗位为 manager的人的信息


发现两者使用重合信息存在的,所以可以使用 union

1.png

使用 union ,会自动去除 两者 重合部分的 重复数据


union all

在union的基础上,不会去除重复部分的重复数据

1.png

重复的数据会被保留


2. 内连接

内连接 是指 使用where子句对两种表形式的笛卡尔积 进行筛选


语法:
select 字段 from 表1 inner join 表2 on 连接条件 and 连接条件;


显示SMITH的名字和部门名称

1.png

筛选条件为 emp.deptno=dept.deptno and ename=‘SMITH’
表示 两张表的 对应 部门号 要相同 ,同时 名字为 SMITH

3. 外连接

左外连接

左侧表完全显示 右侧的表按条件拼接(条件满足拼接,条件不满足拼NULL)


语法:
select 字段名 from 表名1 left join 表名2 on 连接条件

相对于内连接的语法,外连接语法只是把inner 替换成了 left


1.png

创建一张表 stu ,内部包含 id 和 name


1.png

向stu表中 插入对应的数据 ,并 将其显示出来


再次创建一张成绩表 exam ,内部 包含id 和grade


1.png

向exam表中 插入对应的数据,并将其显示出来


1.png

发现stu 学生表中 有四名同学,而exam成绩表中 只有三个成绩
而且成绩表中有一个 不存在的id值为11 的学生


查询所有学生的成绩,如果这个学生没有成绩,也要将学生的个人信息显示出来

1.png

由于有些学生是没有成绩的,就导致两张表中的id 匹配不上
若只拿id 做内连接,就只有 id值为 1 2 的信息能够显示


1.png

为了保证所有学生的信息都显示,所以使用 左外连接

右外连接

左侧的表按条件拼接(条件满足拼接,条件不满足拼NULL) 右侧表完全显示


语法:
select 字段 from 表名1 right join 表名2 on 连接条件


对stu表和exam表联合查询,把所有的成绩都显示出来,即使这个成绩没有学生与它对应,也要
显示出来

1.png

stu 学生表中 有四名同学,而exam成绩表中 只有三个成绩
而且成绩表中有一个 不存在的id值为11 的学生


1.png

此时右侧exam表的三个id值 全部显示出来了
左侧stu表 由于只有两个id值与exam表中的id值对应,所以只显示2个,剩下显示为NULL


猜你喜欢

【MySql】mysql数据库表分区设计实现
文章目录前言一、什么是mysql表分区?二、表分区有哪几种方案? 如何实现1、创建订单表2、添加订单的索引为联合索引(需要修改为索引联合索引,不然后面会报错)3、按照订单时间范围来分区4、向数据库插入数据(创建存储过程,循环插入2023到2025年的数据,数据时间间隔为半个小时1次)5、查询分区数据插入情况6、根据分区查询数据2.1 范围分区(Range Partitioning)2.2 列表分区 (List Partitioning)2.3 哈希分区 (hash Partinioning)2.
发表于:2024-07-16 浏览:376 TAG:
【MySql】select * 查询慢的原因和优化方法
&zwnj;使用SELECT&nbsp;*查询语句会导致查询效率低下的原因主要包括&zwnj;:它会查询所有的列和行数据,包括不需要的和重复的列,因此会占用更多的系统资源,导致查询效率低下。此外,由于...
发表于:2024-10-21 浏览:271 TAG: #mysql
【MySql】深入解析数据库索引
写在前面&nbsp;MySQL索引是数据库中一个关键的概念,它可以极大地提高查询性能,加快数据检索速度。但是,要充分发挥索引的作用,需要深入理解它们的工作原理和使用方式。在本文中,我们将深入解析MySQL索引,探讨它们的重要性、类型、创建、维护以及最佳实践。一、基础介绍&nbsp;1.1&nbsp;什么是索引?在数据库中,索引是一种数据结构,用于快速查找表中的数据。索引包含表中一列或多列的值,这些值按照一定的顺序进行排序,以便优化数据的检索速度。通过使用索引,数据库可以避免全表扫描,从而提高查询
发表于:2023-12-07 浏览:350 TAG:
【MySQL】MySQL数据库CPU飙升到100%解决方案
1、定位cpu问题所在当cpu飙升到100%时,先用操作系统命令top命令观察是不是mysqld占用导致的,如果不是,找出占用高的进程,并进行相关处理。2、查看慢查询日志进入mysql命令行mysql&nbsp;-h主机地址&nbsp;-u用户名&nbsp;-p用户密码1查看慢查询SQL是否启用:ON是开启,OFF是关闭。 show variables like ‘log_slow_queries’;开启慢查询日志 set global log_slow_queries = on;3、使用sho
发表于:2024-07-17 浏览:247 TAG:
【MySQL】表操作
第三篇:表操作#一、什么是表#表相当于文件,表中的一条记录就相当于文件的一行内容,不同的是,表中的一条记录有对应的标题,称为表的字段二、创建表#语法CREATE&nbsp;TABLE&nbsp;表名( 字段名1&nbsp;&nbsp;类型[(宽度)约束条件], 字段名1&nbsp;&nbsp;类型[(宽度)约束条件], 字段名1&nbsp;&nbsp;类型[(宽度)约束条件] )ENGINE=innodb&nbsp;DEFAULT&nbsp;CHARSET&nbsp;utf8; create
发表于:2024-06-21 浏览:272 TAG: #mysql
【MySql】json字段内数据求和
首先,我们需要使用JSON_EXTRACT函数来解析JSON数据。假设我们的数据表名为data_table,JSON数据字段名为json_data,需要查询的字段为field_name。```sql SELECT&nbsp;JSON_EXTRACT(json_data,&nbsp;&#39;$.field_name&#39;)&nbsp;AS&nbsp;extracted_field FROM&nbsp;data_table;###&nbsp;步骤2:使用JSON函数提取数组内字段 如果字段是一
发表于:2024-09-18 浏览:210 TAG: #mysql
【MySQL】索引有哪些优缺点
索引是数据库中用于提高查询性能的重要工具,但它也有一些有点和确定,一下是索引的主要优缺点:优点:&nbsp;&nbsp;&nbsp;&nbsp;1. 加速查询速度最显而易见的有嗲你是加速select查询速度。通过使用索引,数据库引擎能够更快的定位和检索数据,特别是在大型数据集中。&nbsp;&nbsp;&nbsp;&nbsp;2. 排序性能提升索引可以提高排序操作的性能,因为数据库引擎可以按照索引顺序而不是表中实际物理存储顺序来执行排序。&nbsp;&nbsp;&nbsp;&nbsp;3. 唯一
发表于:2024-06-16 浏览:265 TAG:
【MySql】批量替换数据表中某字段的值
系统域名变更,需要把数据库存储的图片、文件等链接域名也替换掉,使用了文字替换的方法,在此记录一下:数据库:mysqlUPDATE&nbsp;user&nbsp;SET&nbsp;headimg&nbsp;=&nbsp;REPLACE(headimg,&nbsp;&#39;a.com&#39;,&nbsp;&#39;b.com&#39;);其中 user 是数据表,headimg是其中的字段,把headimg中&#39;a.com&#39;全部替换成&#39;b.com&#39;,headimg中
发表于:2024-04-02 浏览:328 TAG:
【MySql】where和having的异同?
文章目录面试常考:where与having有什么不同?一个实际查询需求wherehaving怎么正确的使用where和having?面试常考:where与having有什么不同?我们在进行查询的时候,经常需要按照条件对查询结果进行筛选,这就要用到条件语句where和having了。where是直接对表中的字段进行限定来筛选结果,having则需要跟分组关键字group by一起使用,通过对分组字段或分组计算函数进行限定来筛选结果。虽然它们都是对查询进行限定,却有着各自的特点和适用场景。很多时候,
发表于:2023-12-08 浏览:392 TAG:
【MySQL】MySql中的锁(表锁,行锁)
&nbsp; &nbsp; 锁是计算机协调多个进程或纯线程并发访问某一资源的机制。在数据库中,除传统的计算资源(CPU、RAM、I/O)的争用以外,数据也是一种供许多用户共享的资源。如何保证数据并发访问的一致性、有效性是所在有数据库必须解决的一个问题,锁冲突也是影响数据库并发访问性能的一个重要因素。从这个角度来说,锁对数据库而言显得尤其重要,也更加复杂。&nbsp;概述&nbsp; &nbsp; 相对其他数据库而言,MySQL的锁机制比较简单,其最显著的特点是不同的存储引擎支持不同的锁机制。My
发表于:2024-06-16 浏览:233 TAG: