mysql之外鍵

来源:https://www.cnblogs.com/progor/archive/2018/04/12/8799922.html
-Advertisement-
Play Games

本文內容: 什麼是外鍵 外鍵的增加 外鍵的修改和刪除 外鍵的約束模式 首發日期:2018-04-12 什麼是外鍵: 外鍵就是表中存在一個欄位指向另外一個表的主鍵,那麼這個欄位就可以稱為外鍵。 一張表可以有多個外鍵。 外鍵用於約束表與表之間的關係,可以說外鍵是表之間的映射關係,這個關係可以幫助我們處理... ...



本文內容:

 

  • 什麼是外鍵
  • 外鍵的增加
  • 外鍵的修改和刪除
  • 外鍵的約束模式

 

首發日期:2018-04-12


什麼是外鍵:

 

  • 外鍵就是表中存在一個欄位指向另外一個表的主鍵,那麼這個欄位就可以稱為外鍵。
  • 一張表可以有多個外鍵。
  • 外鍵用於約束表與表之間的關係,可以說外鍵是表之間的映射關係,這個關係可以幫助我們處理表之間關係的緊密性和存在性(比如學生表的cid班級號不能為不存在的,如果不增加外鍵cid,沒有建立上關係,我們就不知道班級號不存在。)。
  • 或者說,外鍵是告訴資料庫系統,我們所認為的關係,單純的數據,系統是不知道實際意義的,外鍵就是告訴系統應該如何處理他們的關係。
  • 所以,外鍵的核心是約束。

 

 


外鍵的增加:

 

  • 創建外鍵的前提是該欄位首先是一個索引,如果不是的話,創建外鍵是會創建成一個普通索引【所以可以不在意】。
  • 創建外鍵的另外一個前提是“指向表”已經創建,對於一個不存在的表,將無法使用外鍵對應上。
  • 增加的方式:
    • 1.在創建表的時候定義,在所有欄位定義結束後使用foreign key(外鍵欄位) references 指向表(主鍵)來 定義,比如image
    • 2.也可以修改欄位來增加: alter table 表名 add [constraint 外鍵名字] foreign key(外鍵欄位) references 父表(主鍵欄位);
      • constraint 外鍵名字:可以幫助定義外鍵的名字,但不建議使用,因為要求外鍵名都唯一,而使用系統自定義的絕對不會重覆,

 

create table student(
id int primary key auto_increment,
name varchar(15) not null,
gender varchar(10) not null,
cid int,
foreign key(cid) references class(id)
);
create table class(
id int primary key auto_increment,
cname varchar(15)
);

 

 

補充:

  • 在Mysql中,如果存儲引擎不是innodb,那麼無法使外鍵的約束作用生效,即使是能成功增加外鍵。
  • 外鍵名不能重覆,所以不建議使用constraint 外鍵名字

 

 


外鍵的修改與刪除:

 

  • 修改:不能修改外鍵信息,如外鍵指向之類的,只能先刪除再新增。
  • 刪除語法:alter table 表名 drop foreign key 外鍵名;
    • 這裡的外鍵名不是外鍵欄位,而是外鍵名。如果沒有使用constraint來定義,可以通過show create來查看表創建語句中系統定義的外鍵名。
    • image

 

 

 

補充:

  • 刪除外鍵時,如果使用desc會看到表結構還有MUL,那是一個索引。因為創建外鍵時,欄位會被創建成一個索引。如果不想保留,可以使用drop index 欄位名 on 表名.

 

 

 


外鍵的約束模式:

 

  • 外鍵是用來約束表之間的關係的。
  • (約定創建外鍵的表稱為子表,指向的表稱為父表)
    • 針對子表:可以約束父表的插入和修改【這種約束是父表對子表的約束】
      • 涉及到外鍵的插入和修改時,如果外鍵欄位找不到對應的匹配那麼會插入\修改失敗(像插入選課記錄不可能插入一門課程表中沒有的課程)。
      • 比如:image
    • 針對父表:可以約束父表的刪除和更新,通常有可以一下幾種約束模式。【這種約束是子表對父表的約束】
      • 模式:
        • strict嚴格模式:涉及到外鍵的刪除和更新時,如果對應記錄的主鍵數據已經被子表使用時,那麼無法刪除(像已經有人入學了某個班級,學校不可能犯傻去把某個班級刪除,只能刪除那些沒人入學的班級。)
        • 級聯模式:涉及到外鍵的刪除和更新時,如果欄位已經被子表使用,子表中的數據會對應更新(像某個班改了班號,那麼學生表中的班別都對應更改;如果某個班被刪除,就刪除對應班的所有學生)
        • 置空模式:涉及到外鍵的刪除和更新時,如果欄位已經被子表使用,那麼子表中的外鍵數據會置空(像某個班被刪掉了,不應該刪掉所有學生,而是應該給他們先置空再重新分配班別)【子表允許置空的前提是該欄位允許為空】
      • 其實可以給不同操作指定不同模式
      • 綜上所述(根據我的那些舉例),實際上,合適的舉措是刪除時置空(即使某個班太垃圾了,想刪除某個班,但也不應該將所有學生退學,而是將它們分到別的班),修改時級聯(允許更改班號,而且更改會更新到學生中)
      • 不同操作設置不同模式的設置方法(在子表中操作):foreign key(外鍵欄位) references 父表(主鍵) on 操作 模式
        foreign key(外鍵欄位) references 父表 (主鍵) on delete set null on update cascade;

     

    -- 實驗表結構
    create table class(
    id int primary key auto_increment,
    cname varchar(15)
    );
    create table student2(
    id int primary key auto_increment,
    name varchar(15) not null,
    gender varchar(10) not null,
    cid int,
    foreign key(cid) references class(id) on delete set null on update cascade
    );
    -- 實驗表數據:
    insert into class(cname) values("python"),("linux"),("java"),("html5");
    
    insert into student2(name,gender,cid) values("Alice","female",1);
    insert into student2(name,gender,cid) values("John","female",2);
    insert into student2(name,gender,cid) values("Jack","female",3);
    insert into student2(name,gender,cid) values("Amy","female",4);
    
    select * from student2;
    select * from class;
    -- 嘗試更新級聯
    update class set id = 6 where cname="python";
    select * from student2; -- 結果原來的python的cid=6
    -- 嘗試刪除置空
    delete from class where cname="java";
    select * from student2; -- 結果原來的java的cid=null

     

     

    補充:

    • 需要設置好約束模式,不要在多個子表中使用不同的約束模式,不然會衝突。

     

     



    您的分享是我們最大的動力!

    -Advertisement-
    Play Games
    更多相關文章
    • VMwareTools可以實現在虛擬機與實體機之間自由複製文件 1、點擊菜單欄–>虛擬機–>安裝VMware Tools2、獲取管理員許可權輸入以下命令,來獲取管理員許可權,密碼是你在上面設置的密碼:sudo passwd 3、進入VMware Tools目錄,解壓文件tar -zxvf VMwareT ...
    • nohup -- invoke a utility immune to hangups : 運行命令忽略掛起信號 & 是指後臺運行; nohup 的功能和& 之間的功能並不相同。其中,nohup 可以使得命令永遠運行下去和用戶終端沒有關係。當我們斷開ssh 連接的時候不會影響他的運行。而& 表示後臺 ...
    • 因為今天寫了個小腳本,死活不成功,總是報文件或者目錄不存在,問了一下我們馬同學的正常寫法,發現只有腳本頭的區別,也就是今天本文要講的 !/bin/sh與 !/bin/bash. 本文參考:https://www.cnblogs.com/aaronLinux/p/6885288.html和http:/ ...
    • 今天在linux下創建了一個項目自動化發佈的腳本,在執行到 mvn clean package -Dmaven.test.skip=true 的時候,項目clean失敗 查下下度娘,windows下導致原因說是文件被占用,初步懷疑linux下是許可權不夠導致,使用 sudo -i 切換到root用戶下 ...
    • 修改完nginx配置後,需要使用 nginx -s reload使修改的配置生效,配置生效是平滑的,不會對訪問產生任何影響reload後會啟動新的進程接受新請求,對於未處理完的請求還是用老的配置,直到請求處理完畢,老的進行會退出 ...
    • 本文收錄在Linux運維企業架構實戰系列 環境準備 系統:CentOS 7 IP:192.168.10.101 關閉selinux 和防火牆 一、準備 Python3 和 Python 虛擬環境 1、安裝依賴包 [root@centos7-1 opt]# yum -y install wget sq ...
    • Swap分區的拓展和縮小 分為兩種方法: 一, 用磁碟直接掛在 步驟如下: 使用fdisk來創建交換分區(假設 /dev/sdb2 是創建的交換分區) 使用 mkswap 命令來設置交換分區: # mkswap /dev/sdb2 啟用交換分區: # swapon /dev/sdb2 寫入/etc/ ...
    • revit二次開發中遇到的問題 RevitAPIUI.dll 只能 Native Library 中執行; 脫離了Native Library,API是跑不起來的 。 檢查程式流程:登錄,配置,啟動revit。 在啟動revit之前不能執行RevitAPI.dll和RevitAPIUI.dll的相關 ...
    一周排行
      -Advertisement-
      Play Games
    • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
    • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
    • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
    • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
    • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
    • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
    • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
    • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
    • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
    • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...