Mysql 03: 连接查询全解——内连接、外连接与复合条件查询

在 MySQL 中,连接查询(JOIN) 是多表数据关联查询的核心语法,用于从两张及以上表中提取关联数据。本文围绕图片中的核心知识点,从语法、场景、代码示例三个维度,带你彻底掌握 MySQL 连接查询。


一、核心知识点总览

连接类型 语法 核心逻辑 结果特点
内连接(INNER JOIN) A INNER JOIN B ON 关联条件 只返回两张表中匹配成功的记录 无匹配数据不显示
左连接(LEFT JOIN) A LEFT JOIN B ON 关联条件 左表 A 为基准,匹配右表 B;左表全显示,右表无匹配则补 NULL 左表数据完整,右表按需补空
右连接(RIGHT JOIN) A RIGHT JOIN B ON 关联条件 右表 B 为基准,匹配左表 A;右表全显示,左表无匹配则补 NULL 右表数据完整,左表按需补空
复合条件连接 连接查询 + 多条件过滤 在连接基础上,追加 WHERE/ON 多条件筛选 精准过滤关联数据

二、准备测试数据

先创建两张关联表,用于后续所有示例:

1. 学生表 student(左表)

sql 复制代码
CREATE TABLE score (
    score_id INT PRIMARY KEY AUTO_INCREMENT,
    stu_id INT NOT NULL,
    subject VARCHAR(20) NOT NULL,
    score INT NOT NULL,
    FOREIGN KEY (stu_id) REFERENCES student(stu_id)
);
INSERT INTO score (stu_id, subject, score) VALUES
(1, '语文', 85),
(1, '数学', 92),
(2, '语文', 78),
(3, '数学', 88); -- 赵六(stu_id=4)无成绩

2. 成绩表 score(右表)

sql 复制代码
CREATE TABLE score (
    score_id INT PRIMARY KEY AUTO_INCREMENT,
    stu_id INT NOT NULL,
    subject VARCHAR(20) NOT NULL,
    score INT NOT NULL,
    FOREIGN KEY (stu_id) REFERENCES student(stu_id)
);
INSERT INTO score (stu_id, subject, score) VALUES
(1, '语文', 85),
(1, '数学', 92),
(2, '语文', 78),
(3, '数学', 88); -- 赵六(stu_id=4)无成绩

三、各连接类型详解 + 代码示例

1. 内连接查询(INNER JOIN)

核心逻辑

内连接是最常用的连接方式,只返回两张表中满足关联条件的记录,相当于 "取两张表的交集"。

  • 等价语法:SELECT * FROM A,B WHERE A.id = B.id(隐式内连接)
  • 标准语法:SELECT * FROM A INNER JOIN B ON A.id = B.id(显式内连接,推荐)
代码示例
sql 复制代码
-- 显式内连接:查询有成绩的学生信息+成绩
SELECT 
    s.stu_id, s.stu_name, s.class, 
    sc.subject, sc.score
FROM student s
INNER JOIN score sc 
    ON s.stu_id = sc.stu_id;
运行结果
stu_id stu_name class subject score
1 张三 一班 语文 85
1 张三 一班 数学 92
2 李四 二班 语文 78
3 王五 一班 数学 88
说明
  • 赵六(stu_id=4)无成绩,内连接中不会显示,因为右表无匹配数据。

2. 外连接查询

外连接会以某一张表为基准,保留基准表的所有数据,另一张表无匹配则补 NULL,分为左连接右连接

(1)左连接(LEFT JOIN)
核心逻辑

左表为基准 ,左表所有记录全部显示,右表匹配到则显示数据,匹配不到则补 NULL

  • 语法:A LEFT JOIN B ON 关联条件(A 是左表,B 是右表)
代码示例
sql 复制代码
-- 左连接:以学生表为基准,查询所有学生(含无成绩的赵六)
SELECT 
    s.stu_id, s.stu_name, s.class, 
    sc.subject, sc.score
FROM student s
LEFT JOIN score sc 
    ON s.stu_id = sc.stu_id;
运行结果
stu_id stu_name class subject score
1 张三 一班 语文 85
1 张三 一班 数学 92
2 李四 二班 语文 78
3 王五 一班 数学 88
4 赵六 三班 NULL NULL
说明
  • 左表(学生表)的所有学生都显示,赵六无成绩,右表字段补 NULL

(2)右连接(RIGHT JOIN)
核心逻辑

右表为基准 ,右表所有记录全部显示,左表匹配到则显示数据,匹配不到则补 NULL

  • 语法:A RIGHT JOIN B ON 关联条件(B 是右表,A 是左表)
  • 本质:A LEFT JOIN B 等价于 B RIGHT JOIN A,可互相转换,实际开发中优先用左连接(可读性更强)
代码示例
sql 复制代码
-- 右连接:以成绩表为基准,查询所有成绩(无学生的成绩不会出现,本例无此场景)
SELECT 
    s.stu_id, s.stu_name, s.class, 
    sc.subject, sc.score
FROM student s
RIGHT JOIN score sc 
    ON s.stu_id = sc.stu_id;
运行结果

与内连接结果完全一致(因为成绩表的所有记录都能匹配到学生表)。


3. 复合条件连接查询

核心逻辑

在连接查询的基础上,追加多个过滤条件 ,精准筛选关联数据。条件可写在 ON 子句(连接时过滤)或 WHERE 子句(连接后过滤)。

代码示例 1:连接 + 单条件过滤
sql 复制代码
-- 左连接+WHERE:查询一班所有学生(含无成绩)
SELECT 
    s.stu_id, s.stu_name, s.class, 
    sc.subject, sc.score
FROM student s
LEFT JOIN score sc 
    ON s.stu_id = sc.stu_id
WHERE s.class = '一班';
运行结果
stu_id stu_name class subject score
1 张三 一班 语文 85
1 张三 一班 数学 92
3 王五 一班 数学 88

代码示例 2:多条件复合连接
sql 复制代码
-- 复合条件:左连接+多条件,查询一班成绩>80的学生,无成绩也显示
SELECT 
    s.stu_id, s.stu_name, s.class, 
    sc.subject, sc.score
FROM student s
LEFT JOIN score sc 
    ON s.stu_id = sc.stu_id 
    AND sc.score > 80  -- 连接时过滤成绩>80
WHERE s.class = '一班'; -- 过滤班级
运行结果
stu_id stu_name class subject score
1 张三 一班 数学 92
3 王五 一班 数学 88
关键说明:ONWHERE 的区别
  • ON 子句:连接时过滤,只影响右表的匹配,不影响左表的显示(左表仍全显示)
  • WHERE 子句:连接后过滤,会直接过滤最终结果,不符合条件的行直接删除

四、核心对比与避坑指南

1. 内连接 vs 外连接 核心区别

场景 推荐连接 原因
只需要匹配成功的数据 内连接 结果最精简,性能最优
需要保留左表所有数据 左连接 左表完整,右表补空
需要保留右表所有数据 右连接 可转换为左连接,优先用左连接

2. 常见避坑

1.关联条件缺失 :多表连接必须加 ON 关联条件,否则会产生笛卡尔积(数据爆炸)

sql 复制代码
-- 错误!无关联条件,笛卡尔积
SELECT * FROM student, score;

2.ONWHERE 混用 :外连接中,WHERE 会过滤左表数据,导致外连接失效

sql 复制代码
-- 错误!WHERE sc.score>80 会过滤掉赵六的NULL行,左连接变内连接
SELECT * FROM student s LEFT JOIN score sc ON s.stu_id=sc.stu_id WHERE sc.score>80;

3.表别名不规范:多表同名字段必须加表别名,否则报错

sql 复制代码
-- 错误!stu_id 不明确属于哪个表
SELECT stu_id FROM student s JOIN score sc ON s.stu_id=sc.stu_id;
-- 正确:加表别名
SELECT s.stu_id FROM student s JOIN score sc ON s.stu_id=sc.stu_id;

五、综合实战:多表连接 + 复合条件

sql 复制代码
-- 需求:查询所有学生的总分,无成绩的学生总分显示0,按班级分组
SELECT 
    s.class,
    s.stu_name,
    IFNULL(SUM(sc.score), 0) AS total_score  -- IFNULL 把NULL转为0
FROM student s
LEFT JOIN score sc 
    ON s.stu_id = sc.stu_id
GROUP BY s.stu_id, s.stu_name, s.class
ORDER BY total_score DESC;

运行结果

class stu_name total_score
一班 张三 177
一班 王五 88
二班 李四 78
三班 赵六 0

六、核心总结

  1. 内连接:取交集,只显示匹配数据
  2. 左连接:左表全显,右表补空;右连接:右表全显,左表补空
  3. 复合条件ON 过滤连接,WHERE 过滤结果,不可混用
  4. 避坑关键 :必须加关联条件,同名字段加别名,外连接慎用 WHERE 过滤
  5. 性能优化:大表连接优先用内连接,关联字段加索引,避免笛卡尔积
相关推荐
小文大数据2 小时前
python实现HTML转PDF
java·前端·数据库
Java水解2 小时前
【MYSQL】MYSQL学习的一大重点:MYSQL数据类型
后端·mysql
IvorySQL2 小时前
PostgreSQL 技术日报 (4月2日)|社区工具更新与内核补丁讨论速递
数据库·postgresql·开源
vivo互联网技术2 小时前
CVPR 2026 | C²FG:用分数差异分析提高条件生成中CFG的引导
人工智能·算法·aigc
polaris06302 小时前
完美解决phpstudy安装后mysql无法启动
数据库·mysql
极创信息3 小时前
信创软件安全加固指南,信创软件的纵深防御体系
java·大数据·数据库·金融·php·mvc·软件工程
麦聪聊数据3 小时前
企业数据流通与敏捷API交付实战(一):ETL、CDC与API调用对比
数据库·数据仓库·低代码·restful·etl
笑梦无境3 小时前
mysql基础篇四(多年前整理)
数据库·mysql
yy55273 小时前
Mysql 主从复制与读写分离
android·数据库·mysql