数据库(MySQL)—— 多表查询

数据库(MySQL)—— 多表查询

  • 多表关系
  • 一对多
  • 多对多
  • 一对一
  • 多表查询概述
    • 数据准备
    • 查询形式
    • 笛卡尔积
  • 分类
    • 连接查询
      • 内连接
      • 外连接
        • 左外连接
        • 右外连接
      • 自连接
      • 联合查询

今天我们来进入MySQL中一个非常重要的部分:多表查询

多表关系

多表关系是数据库设计中常见的概念,指的是在关系型数据库中,两个或多个数据表之间存在的关联关系。这些关系可以是一对一(1:1)、一对多(1:N)或多对多(M:N)等类型。多表关系的建立有助于实现数据的规范化存储和高效查询。

  1. 一对一关系(1:1):在这种关系中,一个表中的记录与另一个表中的记录有且仅有一个对应关系。例如,一个学生表(Student)和一个学生详情表(StudentDetail),每个学生都有一个唯一的详情记录,反之亦然。
  2. 一对多关系(1:N):在这种关系中,一个表中的一条记录可以与另一个表中的多条记录相关联。例如,一个班级表(Class)和学生表(Student),一个班级可以有多个学生,但每个学生只能属于一个班级。
  3. 多对多关系(M:N):在这种关系中,一个表中的一条记录可以与另一个表中的多条记录相关联,反之亦然。例如,一个学生表(Student)和一个课程表(Course),一个学生可以选修多门课程,同时一门课程也可以被多个学生选修。为了实现这种关系,通常需要引入一个中间表(如StudentCourse),该表记录学生和课程之间的关联信息。

在数据库设计中,通过定义主键(Primary Key)和外键(Foreign Key)来建立和维护多表关系主键用于唯一标识一个表中的记录,而外键则用于在一个表中引用另一个表的主键。当两个表之间存在关联关系时,可以在一个表的外键列中存储另一个表的主键值,从而实现表的关联查询。

通过合理地设计多表关系,可以提高数据库的查询性能和数据完整性。同时,多表关系也有助于实现数据的逻辑分离和模块化,使得数据库结构更加清晰易懂。

一对多

  • 案例: 部门 与 员工的关系
  • 关系: 一个部门对应多个员工,一个员工对应一个部门
  • 实现: 在多的一方建立外键,指向一的一方的主键

对应的SQL语句如下:

-- 创建emp表
CREATE TABLE emp(-- id号id      int PRIMARY KEY AUTO_INCREMENT COMMENT '员工id',name    VARCHAR(10) UNIQUE COMMENT '员工姓名',age     TINYINT UNSIGNED COMMENT '员工年龄',dept_id TINYINT UNSIGNED COMMENT '部门编号',-- 外键CONSTRAINT fk_key_dept_id FOREIGN KEY(dept_id) REFERENCESdept(id)
)COMMENT '员工表';-- 部门表
CREATE TABLE dept(id TINYINT UNSIGNED PRIMARY KEY  COMMENT '部门编号',name VARCHAR(10) COMMENT '部门名字'
)COMMENT '部门表';

在这里插入图片描述

多对多

  • 案例: 学生 与 课程的关系
  • 关系: 一个学生可以选修多门课程,一门课程也可以供多个学生选择
  • 实现: 建立第三张中间表,中间表至少包含两个外键,分别关联两方主键
CREATE TABLE students(id int UNSIGNED PRIMARY KEY AUTO_INCREMENT COMMENT '序号',name VARCHAR(10) UNIQUE COMMENT '学生姓名',no int UNSIGNED UNIQUE COMMENT '学生学号'
)COMMENT '学生表';-- 选课表
CREATE TABLE courses(id int PRIMARY KEY AUTO_INCREMENT COMMENT '序号',name VARCHAR(10) COMMENT '课程名字'
)COMMENT '选课表';-- 中间表
CREATE TABLE stu_cour(id int PRIMARY KEY AUTO_INCREMENT COMMENT '序号',studentid int UNSIGNED NOT NULL COMMENT '学生ID',courseid int  NOT NULL COMMENT '课程ID',-- 外键CONSTRAINT fk_stu_no FOREIGN KEY (studentid) REFERENCESstudents(id),CONSTRAINT  fk_cour_no FOREIGN KEY (courseid) REFERENCEScourses(id)
)COMMENT '学生选课详情表';

在这里插入图片描述

一对一

  • 案例: 用户 与 用户详情的关系
  • 关系: 一对一关系,多用于单表拆分,将一张表的基础字段放在一张表中,其他详情字段放在另一张表中,以提升操作效率
  • 实现: 在任意一方加入外键,关联另外一方的主键,并且设置外键为唯一的(UNIQUE)
CREATE TABLE tb_user(id int PRIMARY KEY AUTO_INCREMENT COMMENT '序号',age TINYINT UNSIGNED COMMENT '年龄',name VARCHAR(10) NOT NULL COMMENT '姓名',gender char(1) NOT NULL COMMENT '性别',phone char(11) UNIQUE COMMENT '电话号码'
)COMMENT '用户基本信息表';CREATE TABLE tb_user_edu(id int PRIMARY KEY AUTO_INCREMENT COMMENT '序号',degree VARCHAR(10) NOT NULL COMMENT '学历',major VARCHAR(10) NOT NULL COMMENT '学位',primaryschool VARCHAR(20) NOT NULL COMMENT '小学',middleschool VARCHAR(20) NOT NULL COMMENT '中学',university VARCHAR(20) NOT NULL COMMENT '大学',userid int UNIQUE COMMENT '用户编号',CONSTRAINT fk_to_id FOREIGN KEY(userid) REFERENCEStb_user(id) -- 外键
)COMMENT '用户教育信息表';insert into tb_user(id, name, age, gender, phone) values
(null,'黄渤',45,'1','18800001111'),
(null,'冰冰',35,'2','18800002222'),
(null,'码云',55,'1','18800008888'),
(null,'李彦宏',50,'1','18800009999');
insert into tb_user_edu(id, degree, major, primaryschool, middleschool,
university, userid) values
(null,'本科','舞蹈','静安区第一小学','静安区第一中学','北京舞蹈学院',1),
(null,'硕士','表演','朝阳区第一小学','朝阳区第一中学','北京电影学院',2),
(null,'本科','英语','杭州市第一小学','杭州市第一中学','杭州师范大学',3),
(null,'本科','应用数学','阳泉第一小学','阳泉区第一中学','清华大学',4);

在这里插入图片描述

多表查询概述

数据准备

删除之前 emp, dept表的测试数据
执行如下脚本,创建emp表与dept表并插入测试数据

DROP TABLE IF EXISTS emp,dept;
create table dept(id int auto_increment comment 'ID' primary key,name varchar(50) not null comment '部门名称'
)comment '部门表';
INSERT INTO dept (id, name) VALUES (1, '研发部'), (2, '市场部'),(3, '财务部'), (4,
'销售部'), (5, '总经办'), (6, '人事部');-- 创建emp表,并插入数据
create table emp(id int auto_increment comment 'ID' primary key,name varchar(50) not null comment '姓名',age int comment '年龄',job varchar(20) comment '职位',salary int comment '薪资',entrydate date comment '入职时间',managerid int comment '直属领导ID',dept_id int comment '部门ID'
)comment '员工表';-- 添加外键
alter table emp add constraint fk_emp_dept_id foreign key (dept_id) references
dept(id);INSERT INTO emp (id, name, age, job,salary, entrydate, managerid, dept_id)
VALUES
(1, '金庸', 66, '总裁',20000, '2000-01-01', null,5),
(2, '张无忌', 20, '项目经理',12500, '2005-12-05', 1,1),
(3, '杨逍', 33, '开发', 8400,'2000-11-03', 2,1),
(4, '韦一笑', 48, '开发',11000, '2002-02-05', 2,1),
(5, '常遇春', 43, '开发',10500, '2004-09-07', 3,1),
(6, '小昭', 19, '程序员鼓励师',6600, '2004-10-12', 2,1),
(7, '灭绝', 60, '财务总监',8500, '2002-09-12', 1,3),
(8, '周芷若', 19, '会计',48000, '2006-06-02', 7,3),
(9, '丁敏君', 23, '出纳',5250, '2009-05-13', 7,3),
(10, '赵敏', 20, '市场部总监',12500, '2004-10-12', 1,2),
(11, '鹿杖客', 56, '职员',3750, '2006-10-03', 10,2),
(12, '鹤笔翁', 19, '职员',3750, '2007-05-09', 10,2),
(13, '方东白', 19, '职员',5500, '2009-02-12', 10,2),
(14, '张三丰', 88, '销售总监',14000, '2004-10-12', 1,4),
(15, '俞莲舟', 38, '销售',4600, '2004-10-12', 14,4),
(16, '宋远桥', 40, '销售',4600, '2004-10-12', 14,4),
(17, '陈友谅', 42, null,2000, '2011-10-12', 1,null);

查询形式

原来查询单表数据,执行的SQL形式为:select * from emp;

那么我们要执行多表查询,就只需要使用逗号分隔多张表即可,如: select * from emp , dept; 具体的执行结果如下:
在这里插入图片描述
此时,我们看到查询结果中包含了大量的结果集,总共102条记录,而这其实就是员工表emp所有的记录
(17) 与 部门表dept所有记录(6) 的所有组合情况,这种现象称之为笛卡尔积。接下来,就来简单介绍下笛卡尔积。

笛卡尔积

笛卡尔积(Cartesian Product)是数学中的一个概念,它表示两个集合A和B的所有可能的有序对的集合。用符号表示为A × B,其中“×”表示笛卡尔积运算。具体来说,A × B = {(a, b) | a ∈ A 且 b ∈ B}

例如,设集合A = {1, 2},集合B = {a, b, c},则A × B = {(1, a), (1, b), (1, c), (2, a), (2, b), (2, c)}。

在这里插入图片描述
而在多表查询中,我们是需要消除无效的笛卡尔积的,只保留两张表关联部分的数据。
在这里插入图片描述
在SQL语句中,如何来去除无效的笛卡尔积呢? 我们可以给多表查询加上连接查询的条件即可:

select * from emp , dept where emp.dept_id = dept.id;

在这里插入图片描述

分类

连接查询

  • 内连接:相当于查询A、B交集部分数据
  • 外连接
    左外连接:查询左表所有数据,以及两张表交集部分数据
    右外连接:查询右表所有数据,以及两张表交集部分数据
    自连接:当前表与自身的连接查询,自连接必须使用表别名

我们先来介绍内连接:

内连接

在这里插入图片描述
内连接的语法分为两种: 隐式内连接、显式内连接。先来学习一下具体的语法结构。

隐式内连接:

SELECT 字段列表 FROM1 ,2 WHERE 条件 ... ;

显式内连接:

SELECT 字段列表 FROM1 [ INNER ] JOIN2 ON 连接条件 ... ; 1

案例:

A. 查询每一个员工的姓名 , 及关联的部门的名称 (隐式内连接实现)
表结构: emp , dept
连接条件: emp.dept_id = dept.id

select emp.name , dept.name from emp , dept where emp.dept_id = dept.id ;

在这里插入图片描述
我们也可以取别名简化操作:

-- 为每一张表起别名,简化SQL编写
select e.name,d.name from emp e , dept d where e.dept_id = d.id;

在这里插入图片描述
外连接:

B. 查询每一个员工的姓名 , 及关联的部门的名称 (显式内连接实现) — INNER JOIN …ON …
表结构: emp , dept
连接条件: emp.dept_id = dept.id

select emp.name,dept.id
from emp inner join
dept on emp.dept_id = dept.id;

在这里插入图片描述

表的别名:
tablea as 别名1 , tableb as 别名2 ;
tablea 别名1 , tableb 别名2 ;

外连接

在这里插入图片描述
外连接分为两种,分别是:左外连接 和 右外连接。具体的语法结构为:

左外连接

案例:
A. 查询emp表的所有数据, 和对应的部门信息
由于需求中提到,要查询emp的所有数据,所以是不能内连接查询的,需要考虑使用外连接查询。
表结构: emp, dept
连接条件: emp.dept_id = dept.id

SELECT 字段列表 FROM1 LEFT [ OUTER ] JOIN2 ON 条件 ... ;

左外连接相当于查询表1(左表)的所有数据,当然也包含表1和表2交集部分的数据。

select * from
emp left join
dept on emp.dept_id = dept.id;

在这里插入图片描述

右外连接

案例

查询dept表的所有数据, 和对应的员工信息(右外连接)
由于需求中提到,要查询dept表的所有数据,所以是不能内连接查询的,需要考虑使用外连接查询。
表结构: emp, dept
连接条件: emp.dept_id = dept.id

SELECT 字段列表 FROM1 RIGHT [ OUTER ] JOIN2 ON 条件 ...; 

在这里插入图片描述

注意事项:
左外连接和右外连接是可以相互替换的,只需要调整在连接查询时SQL中,表结构的先后顺序就可以了。而我们在日常开发使用时,更偏向于左外连接。

自连接

自连接查询,顾名思义,就是自己连接自己,也就是把一张表连接查询多次。我们先来学习一下自连接的查询语法:

SELECT 字段列表 FROM 表A 别名A JOIN 表A 别名B ON 条件 ... ;

而对于自连接查询,可以是内连接查询,也可以是外连接查询。

案例:

A. 查询员工 及其 所属领导的名字
表结构: emp

select a.name, b.name from emp a,emp b
where a.managerid = b.id;

或者这样也行:

select a.name,b.name from emp a
inner join emp b
on a.managerid = b.id;

在这里插入图片描述

查询所有员工 emp 及其领导的名字 emp , 如果员工没有领导, 也需要查询出来
表结构: emp a , emp b

这个时候我们可以用上左外连接:

select a.name,b.name from emp a
left join emp b
on a.managerid = b.id;

在这里插入图片描述

注意事项:
在自连接查询中,必须要为表起别名,要不然我们不清楚所指定的条件、返回的字段,到底是哪一张表的字段。

联合查询

对于union查询,就是把多次查询的结果合并起来,形成一个新的查询结果集。

SELECT 字段列表 FROM 表A ...
UNION [ ALL ]
SELECT 字段列表 FROM 表B ....;

对于联合查询的多张表的列数必须保持一致,字段类型也需要保持一致。
union all 会将全部的数据直接合并在一起,union 会对合并之后的数据去重。

案例:

A. 将薪资低于 5000 的员工 , 和 年龄大于 50 岁的员工全部查询出来.
当前对于这个需求,我们可以直接使用多条件查询,使用逻辑运算符 or 连接即可。 那这里呢,我们也可以通过union/union all来联合查询.

在这里插入图片描述
union all查询出来的结果,仅仅进行简单的合并,并未去重。

select * from emp where salary < 5000
union
select * from emp where age > 50;

在这里插入图片描述
union 联合查询,会对查询出来的结果进行去重处理。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/6362.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

从零开始学AI绘画,万字Stable Diffusion终极教程(四)

【第4期】图生图 欢迎来到SD的终极教程&#xff0c;这是我们的第四节课 这套课程分为六节课&#xff0c;会系统性的介绍sd的全部功能&#xff0c;让你打下坚实牢靠的基础 1.SD入门 2.关键词 3.Lora模型 4.图生图 5.controlnet 6.知识补充 在前面的课程中&#xff0c;我…

主机win10,VMware 装了ubuntu,ubuntu传文件到主机

亲测可用&#xff0c;1分钟搞定&#xff0c;不能用你打死我 使用 FileZilla 工具互传 FileZilla是一款免费的工具&#xff0c;是基于 FTP 协议进行文件互传的&#xff0c;在传输过程中我们的ubuntu是作为服务器&#xff0c; FileZilla 工具则是作为客户端。 1 ubuntu安装 FTP…

【力扣】203、环形链表 II

142. 环形链表 II 要解决这道题&#xff0c;首先需要对问题进行拆解&#xff1a; 确定链表是否存在环确定环的入口点 如何判断是否存在环呢&#xff1f;这个比较容易想到&#xff0c;使用快慢指针即可判断链表是否存在环。我们定义两个指针&#xff1a; ListNode slow head…

容器组_配置初始化容器

&#x1f4d5;作者简介&#xff1a; 过去日记&#xff0c;致力于Java、GoLang,Rust等多种编程语言&#xff0c;热爱技术&#xff0c;喜欢游戏的博主。 &#x1f4d8;相关专栏Rust初阶教程、go语言基础系列、spring教程等&#xff0c;大家有兴趣的可以看一看 &#x1f4d9;Jav…

SSM整合-前后端分离-项目环境搭建 (上)

整合SSM 项目基础环境搭建项目介绍创建项目项目全局配置web.xmlSpringMVC配置配置Spring和MyBatis, 并完成整合创建表, 使用逆向工程生成Bean, XxxMapper和XxxMapper.xml注意事项和细节说明 实现功能01-搭建Vue前端工程需求分析/图解代码实现搭建Vue前端工程vue3项目目录结构梳…

服务网关GateWay原理

文章目录 自动装配核心类GatewayAutoConfigurationDispatcherHandler请求处理阶段apply方法httpHandler#handle方法WebHandler#handle方法DispatchHanlder#handle方法第一步 getHandler获取请求映射第二步 invokeHandler 请求适配第三步 handleResult请求处理总结 上一篇博文我…

Gitea 上传用户签名

在 Gitea 的用户管理部分&#xff0c;有一个 SSH 和 GPG 的选项。 单击这个选项&#xff0c;可以在选项上添加 Key。 Key 的来源 如是 Windows 的用户&#xff0c;可以选择 Kleopatra 这个软件。 通过这个软件生成的 Key 的界面中有一个导出功能。 单击这个导出&#xff0c;…

OpenWRT有线桥接部署教程

前言 之前咱们讲到OpenWRT部署WAN实现PPPoE拨号上网和自动获取IP模式上网的办法&#xff1a; OpenWRT设置PPPoE拨号教程 OpenWRT设置自动获取IP&#xff0c;作为二级路由器 这一次&#xff0c;咱们尝试用OpenWRT有线桥接上一级路由器的教程。 可能有小伙伴敏锐地发现了&am…

【JVM】GC调优(优化JVM参数)、性能调优

GC调优 GC调优的主要目标是避免由垃圾回收引起程序性能下降。 GC调优的核心指标 垃圾回收吞吐量&#xff1a;执行用户代码时间/&#xff08;执行用户代码时间 GC时间&#xff09;延迟&#xff1a;GC延迟 业务执行时间内存使用量 GC调优步骤 发现问题&#xff1a;通过监控…

论文研读|针对文生图模型的AIGC检测

前言&#xff1a;人工智能生成内容的鉴别&#xff08;AIGC检测&#xff09;算是当前的研究热点之一&#xff0c;本篇文章介绍几篇针对文生图模型的 AIGC 检测相关工作。 相关文章&#xff1a;AIGC溯源相关研究详见此篇文章 目录 1. Towards Universal Fake Image Detectors tha…

OneNote导出白色背景文件时将笔记墨迹转换颜色

今天用OneNote导出笔记时发现在文件上做的黑色墨迹笔记全部转成了白色。推测是因为onenote会根据背景色自动转换黑色和白色的墨迹&#xff0c;但是其他颜色好像导出的时候不会转换。 于是&#xff0c;我们首先要转换背景&#xff0c;将黑色背景转成白色背景&#xff0c; 然后将…

GNU Radio创建FFT、IFFT C++ OOT块

文章目录 前言一、GNU Radio官方FFT弊端二、创建自定义的 C OOT 块1、创建 OOT 模块2、创建 OOT 块3、修改 C 和 CMAKE 文件4、编译及安装 OOT 块 三、测试1、grc 图2、运行结果①、时域波形对比②、频谱图对比 四、资源自取 前言 GNU Radio 自带的 FFT 模块使用起来不是很方便…

OCC笔记:选择TopoDS_Shape顶点、边、面等等

1、通过AIS_InteractiveContext的函数访问当前选择的图形 hAISContext->InitSelected(); hAISContext->MoreSelected(); hAISContext->NextSelected()&#xff1b; hAISContext->SelectedShape()&#xff1b; 其中hAISContext->SelectedShape()通过StdSelect_…

项目管理【环境】过程

系列文章目录 【引论一】项目管理的意义 【引论二】项目管理的逻辑 【环境】概述 【环境】原则 【环境】过程 一、规划和管理项目的合规性 1.1 规划和管理项目的合规性 1.2 确认合规要求 1.3 审计&#xff1a;衡量合规的程度 二、项目管理计划和项目文件 2.1 项目管理计划和…

C语言 联合和枚举

目录 1. 联合体1.1 联合体类型的声明1.2 联合体变量的创建1.3 联合体的特点1.4 联合体在内存中的存储1.5 联合体使用举例 2. 枚举类型2.1 枚举类型的声明2.2 枚举变量的创建和初始化2.3 枚举类型的大小2.4 枚举类型的优点 正文开始 上次我们通过《C语言 结构体详解》学习了结构…

C语言 | Leetcode C语言题解之第66题加一

题目&#xff1a; 题解&#xff1a; /*** Note: The returned array must be malloced, assume caller calls free().*/ int* plusOne(int* digits, int digitsSize, int* returnSize){for(int i digitsSize - 1; i > 0; --i){digits[i] digits[i] 1;//最后元素1判断是不…

笔试狂刷--Day12(模拟 + 链表的公共节点 + dp)

大家好,我是LvZi,今天带来笔试狂刷--Day12(模拟 链表的公共节点 dp) 一.删除公共字符&#xff08;哈希&#xff09; 题目链接:删除公共字符&#xff08;哈希&#xff09; 分析: 分别读取俩个字符串,将第二个字符串存储到set之中,再遍历第一个字符串,删除公共字符 代码: …

《网络安全技术 网络安全众测服务要求》

近日&#xff0c;全国网络安全标准化技术委员会发布《网络安全技术 网络安全众测服务要求》&#xff08;GB/T 43741-2024&#xff0c;以下简称“众测服务要求”&#xff09;&#xff0c;并将在2024年11月1日正式实施。 《众测服务要求》确立了网络安全众测服务的角色及其职责&…

ruoyi漏洞总结

若依识别 黑若依 :icon hash"-1231872293 绿若依 :icon hash"706913071” body" 请通过前端地址访 " body" 认证失败&#xff0c;无法访问系统资源 " 如果页面访问显示不正常&#xff0c;可添加默认访问路径尝试是否显示正常 /login?redi…

Dashboard 介绍

Dashboard 介绍 一、K8S Dashboard简介 简单的说&#xff0c;K8S Dashboard是官方的一个基于WEB的用户界面&#xff0c;专门用来管理K8S集群&#xff0c;并可展示集群的状态。K8S集群安装好后默认没有包含Dashboard&#xff0c;我们需要额外创建它 二、RABC简介 还是那句话&a…