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;

查询结果:

相关推荐
小陈工2 小时前
Python Web开发入门(十七):Vue.js与Python后端集成——让前后端真正“握手言和“
开发语言·前端·javascript·数据库·vue.js·人工智能·python
科技小花7 小时前
数据治理平台架构演进观察:AI原生设计如何重构企业数据管理范式
数据库·重构·架构·数据治理·ai-native·ai原生
一江寒逸7 小时前
零基础从入门到精通MySQL(中篇):进阶篇——吃透多表查询、事务核心与高级特性,搞定复杂业务SQL
数据库·sql·mysql
D4c-lovetrain7 小时前
linux个人心得22 (mysql)
数据库·mysql
阿里小阿希7 小时前
CentOS7 PostgreSQL 9.2 升级到 15 完整教程
数据库·postgresql
荒川之神7 小时前
Oracle 数据仓库雪花模型设计(完整实战方案)
数据库·数据仓库·oracle
做个文艺程序员7 小时前
MySQL安全加固十大硬核操作
数据库·mysql·安全
不吃香菜学java8 小时前
Redis简单应用
数据库·spring boot·tomcat·maven
一个天蝎座 白勺 程序猿8 小时前
Apache IoTDB(15):IoTDB查询写回(INTO子句)深度解析——从语法到实战的ETL全链路指南
数据库·apache·etl·iotdb
不知名的老吴8 小时前
Redis的延迟瓶颈:TCP栈开销无法避免
数据库·redis·缓存