SQL中联表查询深入分析

  • LEFT JOIN 左集合(左表)
  • RIGHT JOIN 右集合(右表)
  • FULL OUTER JOIN 并集
  • JOIN (INNER JOIN) 交集

测试数据表:

sql 复制代码
-- 创建数据库 (可选)
CREATE DATABASE IF NOT EXISTS school_demo;
USE school_demo;

-- 1. 老师表
DROP TABLE IF EXISTS teachers;
CREATE TABLE teachers (
    t_id INT PRIMARY KEY AUTO_INCREMENT,
    t_name VARCHAR(50) NOT NULL,
    subject VARCHAR(50) -- 教授科目
);

-- 2. 课程表 (关联老师)
DROP TABLE IF EXISTS courses;
CREATE TABLE courses (
    c_id INT PRIMARY KEY AUTO_INCREMENT,
    c_name VARCHAR(50) NOT NULL,
    t_id INT, -- 外键,关联老师ID
    credit DECIMAL(3,1) -- 学分
);

-- 3. 学生表
DROP TABLE IF EXISTS students;
CREATE TABLE students (
    s_id INT PRIMARY KEY AUTO_INCREMENT,
    s_name VARCHAR(50) NOT NULL,
    age INT,
    major VARCHAR(50) -- 专业
);

-- 4. 选课记录表 (关联学生和课程)
DROP TABLE IF EXISTS selections;
CREATE TABLE selections (
    id INT PRIMARY KEY AUTO_INCREMENT,
    s_id INT, -- 关联学生
    c_id INT, -- 关联课程
    score INT, -- 分数,NULL表示未考试
    INDEX idx_s (s_id),
    INDEX idx_c (c_id)
);

测试数据:

sql 复制代码
-- 插入老师数据 (5条)
INSERT INTO teachers (t_name, subject) VALUES 
('王老师', '数学'),
('李老师', '英语'),
('赵老师', '计算机'),
('陈老师', '物理'),
('刘老师', '体育');

-- 插入课程数据 (5条,分别对应上面的老师)
INSERT INTO courses (c_name, t_id, credit) VALUES 
('高等数学', 1, 4.0),
('大学英语', 2, 3.0),
('Java编程', 3, 3.5),
('大学物理', 4, 4.0),
('篮球', 5, 1.0);

-- 插入学生数据 (5条)
INSERT INTO students (s_name, age, major) VALUES 
('张三', 20, '计算机'),
('李四', 21, '数学'),
('王五', 20, '物理'),
('赵六', 22, '计算机'),
('钱七', 19, '英语');

-- 插入选课记录数据 (约8条,模拟不同学生选不同课)
INSERT INTO selections (s_id, c_id, score) VALUES 
(1, 1, 85.5), -- 张三选修高等数学
(1, 3, 92.0), -- 张三选修Java编程
(2, 1, 95.0), -- 李四选修高等数学
(2, 2, 88.0), -- 李四选修大学英语
(3, 4, 76.0), -- 王五选修大学物理
(4, 3, 89.0), -- 赵六选修Java编程
(4, 5, 100.0),-- 赵六选修篮球
(5, 2, NULL); -- 钱七选修英语但还未考试(score为NULL)

场景 1:基础联表 (INNER JOIN 或 JOIN) 查询所有选了课的学生姓名、课程名和分数:

sql 复制代码
SELECT 
    s.s_name AS 学生姓名,
    c.c_name AS 课程名,
    sel.score AS 分数
FROM selections sel
JOIN students s ON sel.s_id = s.s_id
JOIN courses c ON sel.c_id = c.c_id;

查询结果:

场景 2:多表联查 (包含老师信息) 查询学生张三选修的课程,以及该课程的任课老师:

sql 复制代码
SELECT 
    s.s_name,
    c.c_name,
    t.t_name AS 老师,
    t.subject
FROM students s
JOIN selections sel ON s.s_id = sel.s_id
JOIN courses c ON sel.c_id = c.c_id
JOIN teachers t ON c.t_id = t.t_id
WHERE s.s_name = '张三';

查询结果

场景 3:左联表 (Left Join) 查询所有学生,以及他们的选课情况(即使没选课的学生也要列出):

sql 复制代码
SELECT 
    s.s_name,
    c.c_name,
    sel.score
FROM students s
LEFT JOIN selections sel ON s.s_id = sel.s_id
LEFT JOIN courses c ON sel.c_id = c.c_id;

查询结果:

相关推荐
jiayou641 小时前
KingbaseES 实战:深度解析数据库对象访问权限管理
数据库
李广坤21 小时前
MySQL 大表字段变更实践(改名 + 改类型 + 改长度)
数据库
爱可生开源社区2 天前
2026 年,优秀的 DBA 需要具备哪些素质?
数据库·人工智能·dba
随逸1772 天前
《从零搭建NestJS项目》
数据库·typescript
加号33 天前
windows系统下mysql多源数据库同步部署
数据库·windows·mysql
シ風箏3 天前
MySQL【部署 04】Docker部署 MySQL8.0.32 版本(网盘镜像及启动命令分享)
数据库·mysql·docker
李慕婉学姐3 天前
Springboot智慧社区系统设计与开发6n99s526(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
数据库·spring boot·后端
百锦再3 天前
Django实现接口token检测的实现方案
数据库·python·django·sqlite·flask·fastapi·pip
tryCbest3 天前
数据库SQL学习
数据库·sql
jnrjian3 天前
ORA-01017 查找机器名 用户名 以及library cache lock 参数含义
数据库·oracle