前面小節介紹了表的設計三范式和單表的查詢,本小節介紹如何將通過多個表進行關聯查詢數據,其中連表查詢包括 LEFT JOIN、INNER JOIN、RIGHT JOIN。前面小節介紹過表的設計,其中有學生信息表 student、教師信息表 teacher、課程表 course、學生選課關聯表 student_course。
1. 插入表測試數據
為了演示方便,分別清空學生信息表 student、教師信息表 teacher、課程表 course、學生選課關聯表 student_course 數據,并分別插入測試數據。
1.1 清空表數據
分別清空學生信息表 student、教師信息表 teacher、課程表 course、學生選課關聯表 student_course 數據:
TRUNCATE TABLE student;
TRUNCATE TABLE teacher;
TRUNCATE TABLE course;
TRUNCATE TABLE student_course;
執行結果如下圖:
1.2 向學生信息表 student 插入數據
INSERT INTO student (name,age,id_number) VALUES
('趙小明',21,'420117199303036666'),
('王小紅',22,'420117199303037777'),
('張小虎',18,'420117199303038888'),
('李小平',19,'420117199303039999'),
('劉美麗',21,'420117199303035555'),
('周杰',22,'420117199303034444'),
('秦小賢',19,'420117199303033333'),
('馬笑',23,'420117199303032222'),
('艾倫',22,'420117199303031111'),
('包小天',20,'420117199303030000');
執行結果如下圖:
Tips:如果上述添加表數據出現報錯,可能是因為之前修改表字段的時候把表的字段長度或者類型更改了,請檢查字段類型和長度是否正確,還可以刪除所有表按照之前建表步驟重新建表。
1.3 向教師信息表 teacher 插入數據
INSERT INTO teacher (name,age,id_number,email) VALUES
('Tom',29,'420117202006040000','tom@qq.com'),
('Jack',30,'420117202006041111','jack@qq.com'),
('Mary',31,'420117202006042222','mary@qq.com'),
('Timo',35,'420117202006043333','timo@qq.com'),
('Faker',38,'420117202006044444','faker@qq.com'),
('Bob',35,'420117202006045555','bob@qq.com'),
('kelly',40,'420117202006046666','kelly@qq.com'),
('Rose',42,'420117202006047777','rose@qq.com'),
('Hale',55,'420117202006048888','hale@qq.com'),
('John',49,'420117202006049999','john@qq.com'),
('Amy',55,'42011720200604888X','amy@qq.com'),
('Judy',49,'42011720200604999X','judy@qq.com');
執行結果如下圖:
1.4 向課程表 course 插入數據
因為表 course 有 teacher_id 字段,所以插入數據之前,需要獲取 teacher 表數據:
SELECT * FROM teacher;
執行結果如下圖:
teacher 表 id 的值如上圖所示,對應 course 表 teacher_id,按照上面 id 可插入相關課程信息:
INSERT INTO course (course_name,teacher_id) VALUES
('高等數學',1),
('英語',2),
('政治',3),
('信息論',4),
('數據結構和算法',5),
('體育',6),
('模擬電路',7),
('數字電路',8),
('通信原理',9),
('信號系統',10),
('概率論',13),
('光學原理',14);
執行結果如下圖:
1.5 向學生選課關聯表 student_course 插入數據
INSERT INTO student_course (student_id,course_id) VALUES
(1,1),
(1,2),
(2,3),
(2,4),
(3,5),
(3,6),
(4,7),
(4,8),
(5,9),
(5,10),
(6,1),
(6,2),
(7,3),
(7,4);
執行結果如下圖:
2.LEFT JOIN 左連接
以課程表 course 和 teacher 左連接為例:
SELECT c.id AS course_id,c.*,t.* FROM course c LEFT JOIN teacher t ON c.teacher_id=t.id;
執行結果如下圖:
Tips:這對上述 sql 語句說明如下:
- c.id AS course_id 表示將 course表 中 id 字段重命名為 course_id 展示,其目的是為了防止和 teacher表 中 id 字段混淆;
- c.* 表示 course 表所有字段數據;
- t.* 表示 teacher 表字段所有數據;
- ON 后面跟著的條件是連接表的條件;
- course c 表示將 course 簡寫為 c, teacher t 表示將 teacher 簡寫為 t;
- LEFT JOIN 為左連接,是以左邊的表為’基準’,若右表沒有對應的值,用 NULL 來填補。
3.INNER JOIN 內連接
同樣以表 course 和 teacher 內連接為例:
SELECT c.id AS course_id,c.*,t.* FROM course c INNER JOIN teacher t ON c.teacher_id=t.id;
執行結果如下圖:
Tips:INNER JOIN 為內連接,展示的是左右兩表都有對應的數據。
4.RIGHT JOIN 右連接
同樣以表 course 和 teacher 右連接為例:
SELECT c.id AS course_id,c.*,t.* FROM course c RIGHT JOIN teacher t ON c.teacher_id=t.id;
執行結果如下圖:
Tips:RIGHT JOIN 為右連接,是以右邊的表為’基準’,若左表沒有對應的值,用 NULL 來填補。
5. 多表混合連接
以本小節所有數據全部連接查詢為例:
SELECT * FROM
student a
LEFT JOIN
student_course b
ON a.id=b.student_id
RIGHT JOIN
course c
ON b.course_id=c.id
INNER JOIN teacher d
ON c.teacher_id=d.id;
執行結果如下圖:
Tips:多表混合連接查詢時,后面可以把前面執行的結果集整體當成一個表,例如 SELECT * FROM student a LEFT JOIN student_course b ON a.id=b.student_id RIGHT JOIN course c ON b.course_id=c.id 可以理解為 student 和 student_course 左連接查詢之后的結果集再對 course 右連接查詢。
6. 小結
本小節介紹了如何使用 LEFT JOIN、INNER JOIN、RIGHT JOIN 對表數據進行連接查詢,需要注意的是,ON 后面的表示對連表條件,并且還能對表連接查詢的結果集用 WHERE 進行篩選,例如:
SELECT * FROM
student a
LEFT JOIN
student_course b
ON a.id=b.student_id
RIGHT JOIN
course c
ON b.course_id=c.id
INNER JOIN teacher d
ON c.teacher_id=d.id
WHERE a.age > 18 AND d.age < 55;
代碼塊1234567891011
以上 sql 語句后面使用了 WHERE 條件篩選,表示學生年齡大于 18,教師年齡小于 55。