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
    • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
    • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
    • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
    • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
    • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
    • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
    • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
    • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
    • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
    • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...