SQL 中多表查詢的常見連接方式詳解
以下是用圖表結(jié)合代碼解釋 SQL 中多表查詢的常見連接方式:
一、連接類型圖表(ASCII 形式)
1. 內(nèi)連接 (INNER JOIN) +-------------+ +-------------+ | Table A | | Table B | +-------------+ +-------------+ | 交集部分 | +-----------+ 2. 左連接 (LEFT JOIN) +-------------+ +-------------+ | Table A | | Table B | +-------------+ +-------------+ | 全部保留 | | 右表匹配 | 3. 右連接 (RIGHT JOIN) +-------------+ +-------------+ | Table A | | Table B | +-------------+ +-------------+ | 左表匹配 | | 全部保留 | 4. 全外連接 (FULL OUTER JOIN) +-------------+ +-------------+ | Table A | | Table B | +-------------+ +-------------+ | 全部保留 | | 全部保留 |
二、前置代碼(創(chuàng)建示例表)
-- 創(chuàng)建示例表 CREATE TABLE employees ( id INT PRIMARY KEY, name VARCHAR(50), dept_id INT ); CREATE TABLE departments ( id INT PRIMARY KEY, dept_name VARCHAR(50) ); -- 插入測試數(shù)據(jù) INSERT INTO employees VALUES (1, 'Alice', 101), (2, 'Bob', 102), (3, 'Charlie', 103), (4, 'David', NULL); INSERT INTO departments VALUES (101, 'HR'), (102, 'Engineering'), (104, 'Marketing');
三、連接方式代碼示例
1. 內(nèi)連接(INNER JOIN)
SELECT e.name AS 員工姓名, -- 從 employees 表中選擇員工姓名 d.dept_name AS 部門名稱 -- 從 departments 表中選擇部門名稱 FROM employees e -- 主表:employees(別名為 e) INNER JOIN departments d -- 連接表:departments(別名為 d) ON e.dept_id = d.id; -- 連接條件:員工的部門ID = 部門的ID
分步圖解
假設(shè)表中數(shù)據(jù)如下(基于你之前創(chuàng)建的測試數(shù)據(jù)):
表 employees
id | name | dept_id |
---|---|---|
1 | Alice | 101 |
2 | Bob | 102 |
3 | Charlie | 103 |
4 | David | NULL |
表 departments
id | dept_name |
---|---|
101 | HR |
102 | Engineering |
104 | Marketing |
連接過程
employees(e) departments(d) +----+---------+---------+ +-----+-------------+ | id | name | dept_id | | id | dept_name | +----+---------+---------+ +-----+-------------+ | 1 | Alice | 101 | | 101 | HR | | 2 | Bob | 102 | | 102 | Engineering | | 3 | Charlie | 103 | | 104 | Marketing | | 4 | David | NULL | +-----+-------------+ +----+---------+---------+ 通過 INNER JOIN 連接條件 e.dept_id = d.id:
- Alice(dept_id=101)→ 匹配到 d.id=101(HR)→ 保留記錄
- Bob(dept_id=102)→ 匹配到 d.id=102(Engineering)→ 保留記錄
- Charlie(dept_id=103)→ departments 表中無 id=103 → 排除
- David(dept_id=NULL)→ 無法匹配 → 排除
- Marketing(id=104)→ employees 表中無 dept_id=104 → 排除
關(guān)鍵概念解釋
1. INNER JOIN
的作用
- 僅保留兩個表中匹配的記錄
- 如果員工的
dept_id
在departments
表中找不到對應(yīng)的id
,該員工會被排除 - 如果部門在
employees
表中無人歸屬,該部門也會被排除
2. 為什么用 e.dept_id = d.id
?
- 這是兩個表之間的關(guān)聯(lián)關(guān)系:員工的部門編號(
dept_id
)必須等于部門的主鍵(id
)
3. 別名(e
和 d
)的作用
- 簡化表名書寫(
employees e
= 給表起別名e
) - 明確字段來源(例如
e.name
表示來自員工表,d.dept_name
表示來自部門表)
結(jié)果輸出
員工姓名 | 部門名稱 |
---|---|
Alice | HR |
Bob | Engineering |
結(jié)果說明
- Charlie 被排除:因為
dept_id=103
在departments
表中不存在 - David 被排除:因為
dept_id=NULL
無法匹配任何部門 - Marketing 部門 被排除:因為沒有員工的
dept_id=104
2. 左連接(LEFT JOIN)
SELECT e.name, d.dept_name FROM employees e LEFT JOIN departments d ON e.dept_id = d.id;
結(jié)果: 包含所有左表記錄 + 右表匹配結(jié)果(David 的部門為 NULL)
3. 右連接(RIGHT JOIN)
SELECT e.name, d.dept_name FROM employees e RIGHT JOIN departments d ON e.dept_id = d.id;
結(jié)果: 包含所有右表記錄 + 左表匹配結(jié)果(Marketing 部門無員工)
4. 全外連接(FULL OUTER JOIN)
-- MySQL 不支持 FULL OUTER JOIN,需用 UNION 模擬 SELECT e.name, d.dept_name FROM employees e LEFT JOIN departments d ON e.dept_id = d.id UNION SELECT e.name, d.dept_name FROM employees e RIGHT JOIN departments d ON e.dept_id = d.id;
結(jié)果: 包含所有記錄(含 NULL 匹配)
5. 交叉連接(CROSS JOIN)
SELECT e.name, d.dept_name FROM employees e CROSS JOIN departments d;
結(jié)果: 笛卡爾積(4 員工 × 3 部門 = 12 條記錄)
四、結(jié)果對比表
連接類型 | 結(jié)果記錄數(shù) | 包含 NULL 情況 |
---|---|---|
INNER JOIN | 2 | 無 |
LEFT JOIN | 4 | 右表可能為 NULL |
RIGHT JOIN | 3 | 左表可能為 NULL |
FULL OUTER JOIN | 5 | 兩側(cè)均可能為 NULL |
CROSS JOIN | 12 | 無關(guān)聯(lián)條件,純組合 |
五、關(guān)鍵點總結(jié)
- INNER JOIN 僅保留匹配記錄
- LEFT/RIGHT JOIN 保留一側(cè)全部記錄
- FULL OUTER JOIN 保留兩側(cè)所有記錄
- CROSS JOIN 生成笛卡爾積
- 處理 NULL 值時需注意
COALESCE()
或IFNULL()
函數(shù)的使用
可以通過實際運(yùn)行這些 SQL 語句觀察不同連接方式的差異。
PS、自連接示意圖(ASCII 形式)
自連接 (SELF JOIN) +-------------+ +-------------+ | Table | | Table | | (別名為A) | | (別名為B) | +-------------+ +-------------+ | 關(guān)聯(lián)自身的字段 | +----------------+
典型場景:員工表查詢上下級關(guān)系、分類層級關(guān)系等
二、前置代碼(創(chuàng)建帶層級關(guān)系的表)
-- 創(chuàng)建帶 manager_id 的員工表 CREATE TABLE employees ( id INT PRIMARY KEY, name VARCHAR(50), manager_id INT ); -- 插入測試數(shù)據(jù) INSERT INTO employees VALUES (1, 'Alice', NULL), -- 頂級管理者 (2, 'Bob', 1), -- 向 Alice 匯報 (3, 'Charlie', 1), -- 向 Alice 匯報 (4, 'David', 2); -- 向 Bob 匯報
三、自連接代碼示例
1. 查詢員工及其上級(左連接版)
SELECT e.name AS employee, m.name AS manager FROM employees e LEFT JOIN employees m ON e.manager_id = m.id;
結(jié)果:
+----------+----------+
| employee | manager |
+----------+----------+
| Alice | NULL | -- 沒有上級
| Bob | Alice |
| Charlie | Alice |
| David | Bob |
+----------+----------+
2. 僅查詢有上級的員工(內(nèi)連接版)
SELECT e.name AS employee, m.name AS manager FROM employees e INNER JOIN employees m ON e.manager_id = m.id;
結(jié)果:
+----------+---------+
| employee | manager |
+----------+---------+
| Bob | Alice |
| Charlie | Alice |
| David | Bob |
+----------+---------+
四、結(jié)果對比表
連接方式 | 結(jié)果記錄數(shù) | 包含 NULL 情況 | 典型用途 |
---|---|---|---|
自連接-左連接 | 4 | 頂級管理者的上級為 NULL | 顯示完整層級結(jié)構(gòu) |
自連接-內(nèi)連接 | 3 | 無 NULL | 僅顯示有上下級關(guān)系的記錄 |
五、關(guān)鍵點總結(jié)
- 自連接本質(zhì):將同一張表視為兩個邏輯表進(jìn)行連接
- 必須使用別名:區(qū)分不同角色的表實例(如
e
為員工,m
為上級) - 常用場景:
- 組織結(jié)構(gòu)查詢(員工 ↔ 上級)
- 分類層級(父分類 ↔ 子分類)
- 路徑分析(如路線 A → B → C)
4.NULL 處理:
SELECT e.name, COALESCE(m.name, '頂級管理者') AS manager FROM employees e LEFT JOIN employees m ON e.manager_id = m.id;
到此這篇關(guān)于SQL 中多表查詢的常見連接方式詳解的文章就介紹到這了,更多相關(guān)sql多表查詢連接方式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
深入SQLServer中ISNULL與NULLIF的使用詳解
本篇文章是對SQLServer中ISNULL與NULLIF的使用進(jìn)行了詳細(xì)分析介紹,需要的朋友參考下2013-06-06SQL Server 中查看SQL句子執(zhí)行所用的時間
在MSSQL Server中通過查看SQL語句執(zhí)行所用的時間,來衡量SQL語句的性能。2009-07-07高并發(fā)系統(tǒng)數(shù)據(jù)冪等的解決方案
本文主要介紹高并發(fā)系統(tǒng)數(shù)據(jù)冪等解決方案,這里整理了幾種方案供大家參考,有需要的小伙伴可以參考下2016-08-08