SQL 外鍵Foreign Key全解析
1. 什么是外鍵??? ??
- 定義??:外鍵是數(shù)據(jù)庫表中的一列(或一組列),用于??建立兩個表之間的關(guān)聯(lián)關(guān)系??。外鍵的值必須匹配另一個表的主鍵(Primary Key)或唯一約束(Unique Constraint)的值。
- ??作用??:
- 確保數(shù)據(jù)的??引用完整性??(Referential Integrity),防止無效數(shù)據(jù)插入。
- 維護(hù)表之間的邏輯關(guān)系(如“一對多”或“多對多”)。
??2. 外鍵的語法??
在創(chuàng)建表時定義外鍵:
CREATE TABLE 子表 (
列1 數(shù)據(jù)類型,
列2 數(shù)據(jù)類型,
...
FOREIGN KEY (外鍵列) REFERENCES 父表(主鍵列)
[ON DELETE 約束行為] [ON UPDATE 約束行為]
);在已有表中添加外鍵:
ALTER TABLE 子表 ADD CONSTRAINT 約束名稱 FOREIGN KEY (外鍵列) REFERENCES 父表(主鍵列) [ON DELETE 約束行為] [ON UPDATE 約束行為];
??3. 外鍵的約束行為??
當(dāng)父表的記錄被刪除或更新時,子表的外鍵如何處理?通過 ON DELETE 和 ON UPDATE 指定:
| 約束行為 | 說明 |
|---|---|
| ??CASCADE?? | 級聯(lián)操作。父表刪除/更新記錄時,子表關(guān)聯(lián)記錄也被刪除/更新。 |
| ??SET NULL?? | 父表刪除/更新記錄時,子表的外鍵列設(shè)為 NULL(要求外鍵列允許 NULL)。 |
| ??NO ACTION?? | 默認(rèn)行為。阻止父表的刪除/更新操作,如果子表存在關(guān)聯(lián)記錄。 |
| ??RESTRICT?? | 類似 NO ACTION,立即檢查約束。 |
| ??SET DEFAULT?? | 父表刪除/更新記錄時,子表的外鍵設(shè)為默認(rèn)值(需定義默認(rèn)值)。 |
??4. 多列外鍵??
外鍵可以由多個列組成,需滿足:
- 子表和父表的列數(shù)、順序、數(shù)據(jù)類型一致。
- 父表的列必須有唯一約束(如主鍵或唯一索引)。
??示例??:
CREATE TABLE 訂單詳情 (
訂單ID INT,
產(chǎn)品ID INT,
數(shù)量 INT,
PRIMARY KEY (訂單ID, 產(chǎn)品ID),
FOREIGN KEY (訂單ID) REFERENCES 訂單(訂單ID),
FOREIGN KEY (產(chǎn)品ID) REFERENCES 產(chǎn)品(產(chǎn)品ID)
);??5. 外鍵的限制與注意事項?? ??
- 父表必須有主鍵或唯一約束??。
- ??外鍵列的數(shù)據(jù)類型必須與父表主鍵一致??。
- ??引擎支持??:如 MySQL 的 InnoDB 支持外鍵,而 MyISAM 不支持。
- ??性能影響??:外鍵會增加數(shù)據(jù)操作的檢查開銷,但能提升數(shù)據(jù)一致性。
- ??循環(huán)依賴??:避免兩個表互相引用。
??6. 實際應(yīng)用示例??
??場景??:學(xué)生表(students)和課程表(courses),通過選課表(enrollments)關(guān)聯(lián)。
-- 父表:學(xué)生表
CREATE TABLE students (
student_id INT PRIMARY KEY,
name VARCHAR(50)
);
-- 父表:課程表
CREATE TABLE courses (
course_id INT PRIMARY KEY,
course_name VARCHAR(50)
);
-- 子表:選課表(含外鍵)
CREATE TABLE enrollments (
student_id INT,
course_id INT,
enrollment_date DATE,
FOREIGN KEY (student_id) REFERENCES students(student_id) ON DELETE CASCADE,
FOREIGN KEY (course_id) REFERENCES courses(course_id) ON DELETE RESTRICT
);??插入數(shù)據(jù)??:
-- 插入學(xué)生和課程 INSERT INTO students VALUES (1, 'Alice'); INSERT INTO courses VALUES (101, 'Math'); -- 合法插入:學(xué)生和課程存在 INSERT INTO enrollments VALUES (1, 101, '2023-10-01'); -- 非法插入:學(xué)生不存在,觸發(fā)外鍵錯誤 INSERT INTO enrollments VALUES (999, 101, '2023-10-01'); -- 報錯!
??7. 常見問題??
??外鍵必須指向主鍵嗎???
不,可以指向父表的唯一約束(Unique Constraint)。
??能否跨數(shù)據(jù)庫引用???
通常不支持,外鍵需在同一數(shù)據(jù)庫內(nèi)。
??外鍵是否允許 NULL???
如果外鍵列允許 NULL,則插入 NULL 是合法的(表示無關(guān)聯(lián))。
??如何查看外鍵約束???
使用數(shù)據(jù)庫工具或查詢元數(shù)據(jù)(如 MySQL 的 SHOW CREATE TABLE)。
??8. 總結(jié)?? ??
- 外鍵的核心作用??:維護(hù)數(shù)據(jù)的一致性和關(guān)聯(lián)性。??
- 適用場景??:需要強(qiáng)數(shù)據(jù)完整性的系統(tǒng)(如電商、金融)。??
- 慎用場景??:高并發(fā)寫入且對性能要求極高的系統(tǒng)(需權(quán)衡一致性與性能)。
到此這篇關(guān)于SQL 外鍵(Foreign Key)詳細(xì)講解的文章就介紹到這了,更多相關(guān)SQL 外鍵Foreign Key內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
sql server 2012 數(shù)據(jù)庫所有表里查找某字符串的方法
此TSQL語句是針對SQL Server 2012編寫。如果使用之前版本,需要對部分語句進(jìn)行重寫。2013-03-03
Sql Server 壓縮數(shù)據(jù)庫日志文件的方法
Sql Server 日志 _log.ldf文件太大,數(shù)據(jù)庫文件有500g,日志文件也達(dá)到了500g,占用磁盤空間過大,且可能影響程序性能,需要壓縮日志文件,下面小編給大家講解下Sql Server 壓縮數(shù)據(jù)庫日志文件的方法,感興趣的朋友一起看看吧2022-11-11
SQL Server誤區(qū)30日談 第16天 數(shù)據(jù)的損壞和修復(fù)
我已經(jīng)聽過很多關(guān)于數(shù)據(jù)修復(fù)可以做什么、不可以做什么、什么會導(dǎo)致數(shù)據(jù)損壞以及損壞是否可以自行消失。其實我已經(jīng)針對這類問題寫過多篇博文,因此本篇博文可以作為“流言終結(jié)者”來做一個總結(jié),希望你能有收獲2013-01-01
SQL Server 2012 身份驗證(Authentication)
這篇SQL Server安全文章,我們學(xué)習(xí)了SQL Server里的多個驗證選項。Windows集成身份驗證是最安全的,但并不是都是可行的,微軟多年來已經(jīng)讓SQL Server驗證更加安全。2016-04-04
sqlserver通用的刪除服務(wù)器上的所有相同后綴的臨時表
因為項目需要,必須使用大量臨時表,因此編寫批量刪除臨時表的通用語句2012-05-05

