亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 數據庫 > MySQL > 正文

詳解MySQL中的外鍵約束問題

2024-07-24 13:07:37
字體:
來源:轉載
供稿:網友

這篇文章主要介紹了詳解MySQL中的外鍵約束問題,針對在MySQL中使用InnoDB表的情況,需要的朋友可以參考下

使用MySQL開發過數據庫驅動的小型web應用程序的人都知道,對關系數據庫的表進行創建、檢索、更新和刪除等操作都是些比較簡單的過程。理論上,只要掌握了最常見的SQL語句的用法,并熟悉您選擇使用的服務器端腳本語言,就足以應付對MySQL表所需的各種操作了,尤其是當您使用了快速MyISAM數據庫引擎的時候。但是,即使在最簡單的情況下,事情也要比我們想象的要復雜得多。下面我們用一個典型的例子進行說明。假設您正在運行一個博客網站,您幾乎天天更新,并且該站點允許訪問者評論您的帖子。

在這種情況下,我們的數據庫模式至少應該包括兩個MyISAM表,一個用于存放您的博客文章,另一個來處理訪問者的評論。很明顯,這兩個表之間存在一個一對多的關系,所以我們要在第二個表中定義一個外鍵,以便在更新或者刪除數據行時可以保持數據庫的完整性。

像上面這樣的應用程序,不僅維護兩個表的完整性是一個嚴峻的挑戰,而最大的難點在于我們必須在應用程序級別來維護它們的完整性。這是大部分不要求使用事務的web項目在開發期間所采取的方法,因為MyISAM表可以提供出色的性能。

當然,這樣做也是有代價的,正如我前面所說的,應用程序必須維護數據庫的完整性和一致性,這就意味著要實現更復雜的程序設計邏輯來處理各個表之間的關系。雖然可以通過使用抽象層和ORM模塊來簡化數據庫訪問,但是隨著應用程序所需數據表的數量的增加,處理它們所需的邏輯無疑也會隨之變得越發復雜。

那么,對于MySQL來說,有沒有數據庫級別的外鍵處理方式來幫助維護數據庫完整性的呢? 幸運的是,答案是肯定的!MySQL還可以支持InnoDB表,使我們可以通過一種非常簡單的方式來處理外鍵約束。這個特性允許我們可以觸發器某些動作,諸如更新和刪掉表中的某些數據行以維護預定義的關系。

凡事有利皆有弊,使用InnoDB表的主要缺點是它們的速度要比MyISAM慢,尤其是在必須查詢許多表的大規模應用程序中,這一點尤為明顯。好在較新版本MySQL的MyISAM表也已支持外鍵約束。

本文將介紹如何將外鍵約束應用于InnoDB表。此外,我們還將使用一個簡單的基于PHP的MySQL抽象類來創建有關的示例代碼;當然,您也可以使用自己喜歡的其它服務器端語言?,F在,我們開始介紹如何將外鍵約束應用于MySQL。

使用外鍵約束的時機

老實說,在MySQL中使用InnoDB表的時候,不一定非用外鍵約束不可,然而,為了外鍵約束在某些情況下的功用,我們將通過前面提到的例子的代碼進行具體說明。它包括兩個MyISAM表,分別用于存放博客文章和評論。

定義數據庫模式時,我們要在這兩個表之間建立起一對多的關系,方法是在存放評論的表中創建一個外鍵,以將其中的數據行(即評論)對應到特定的博客文章。下面是創建示例MyISAM表的基本SQL代碼:

 

 
  1. DROP TABLE IF EXISTS `test`.`blogs`; 
  2.  
  3. CREATE TABLE `test`.`blogs` ( 
  4.  
  5. `id` INT(10) UNSIGNED AUTO_INCREMENT, 
  6.  
  7. `title` TEXT, 
  8.  
  9. `content` TEXT, 
  10.  
  11. `author` VARCHAR(45) DEFAULT NULL, 
  12.  
  13. PRIROSE KEY (`id`) 
  14.  
  15. ) ENGINE=MyISAM DEFAULT CHARSET=utf8; 
  16.  
  17.  
  18. DROP TABLE IF EXISTS `test`.`comments`; 
  19.  
  20. CREATE TABLE `test`.`comments` ( 
  21.  
  22. `id` INT(10) UNSIGNED AUTO_INCREMENT, 
  23.  
  24. `blog_id` INT(10) UNSIGNED DEFAULT NULL, 
  25.  
  26. `comment` TEXT, 
  27.  
  28. `author` VARCHAR(45) DEFAULT NULL, 
  29.  
  30. PRIROSE KEY (`id`) 
  31.  
  32. ) ENGINE=MyISAM DEFAULT CHARSET=utf8; 

上面,我們只是定義了兩個MyISAM表,它們構成了博客應用程序的數據層。如您所見,第一個表名為blogs,它由一些含義很明顯的字段組成,分別用于存放每篇博客文章的ID、標題和內容,最后是作者。第二個表名為comments,用于存放各篇博客文章的有關評論,它將博客文章的ID作為它的外鍵,從而建立起一對多的關系。

迄今為止,我們的工作還算輕松,因為我們只是創建了兩個簡單的MyISAM表。下一步,我們要做的是使用一些記錄來填充這些表,以便進一步演示在第一個表中刪除表項時,應該在另一個表中執行那些操作。

更新并維護數據庫的完整性

前面部分,我們創建了兩個MyISAM表,來充當博客應用程序的數據層。當然,上面的介紹還很簡單,我們需要做進一步的討論。為此,我們將向這些表中填入一些記錄,方法是使用SQL命令,具體如下所示:

 

 
  1. INSERT INTO blogs (id, title, content, author) VALUES (NULL,'Title of the first blog entry''Content of the first blog entry''Ian'
  2.  
  3. INSERT INTO comments (id, blog_id, comment, author) VALUES (NULL, 1, 'Commenting first blog entry''Susan Norton'), (NULL, 1, 'Commenting first blog entry''Rose Wilson'

上面的代碼,實際上模擬了讀者Susan和Rose對我們的第一篇博客作出了評論的情況。假設現在我們要用另一篇文章來更新第一篇博客。當然,這種情況是有可能發生的。

在這種情況下,為了維護數據庫的一致性,comments表也必須進行相應的更新,要么通過手工方式更新,或者通過處理數據層的應用程序進行更新。就本例而言,我們將使用SQL命令來完成更新,具體如下所示:

 

 
  1. UPDATE blogs SET id = 2, title = "Title of the first blog entry", content = 'Content of the first blog entry', author = 'John Doe' WHERE id = 1 
  2.  
  3. UPDATE comments SET blog_id = 2 WHERE blod_id = 1 

如前所述,因為第一篇博客的數據項的內容已經更新,所以comments表也必須反映出此變化才行。當然,現實中這個更新操作應該在應用程序層完成,而非手工進行,這就意味著這個邏輯必須使用服務器端語言來實現。

為了完成這個操作,對于PHP來說可以通過一個簡單的子過程即可,但是實際上,如果使用了外鍵約束的話,對comments表的更新操作完全可以委托給數據庫。

就像文章前面所說的那樣,InnoDB MySQL表對這個功能提供了無縫地支持。所以,后面部分我們會使用外鍵約束重新前面的示例代碼。

數據庫的級聯更新

下面,我們將利用外鍵約束和InnoDB表(而非默認的MyISAM類型)來重新構建前面的示例代碼。為此,首先要重新定義這兩個示例表,以便它們可以使用特定的數據庫引擎。為此,可以使用如下所示的SQL代碼:

 

 
  1. DROP TABLE IF EXISTS `test`.`blogs`; 
  2.  
  3. CREATE TABLE `test`.`blogs` ( 
  4.  
  5. `id` INT(10) UNSIGNED AUTO_INCREMENT, 
  6.  
  7. `title` TEXT, 
  8.  
  9. `content` TEXT, 
  10.  
  11. `author` VARCHAR(45) DEFAULT NULL, 
  12.  
  13. PRIROSE KEY (`id`) 
  14.  
  15. ) ENGINE=InnoDB DEFAULT CHARSET=utf8; 
  16.  
  17.  
  18. DROP TABLE IF EXISTS `test`.`comments`; 
  19.  
  20. CREATE TABLE `test`.`comments` ( 
  21.  
  22. `id` INT(10) UNSIGNED AUTO_INCREMENT, 
  23.  
  24. `blog_id` INT(10) UNSIGNED DEFAULT NULL, 
  25.  
  26. `comment` TEXT, 
  27.  
  28. `author` VARCHAR(45) DEFAULT NULL, 
  29.  
  30. PRIROSE KEY (`id`), 
  31.  
  32. KEY `blog_ind` (`blog_id`), 
  33.  
  34. CONSTRAINT `comments_ibfk_1` FOREIGN KEY (`blog_id`) REFERENCES `blogs` (`id`) ON UPDATE CASCADE 
  35.  
  36. ) ENGINE=InnoDB DEFAULT CHARSET=utf8; 

這里的代碼與之前的代碼相比,一個明顯的不同之處在于現在的這兩個表使用了InnoDB存儲引擎,所以能夠支持外鍵約束。除此之外,我們還需要注意定義comments表的代碼:

 

 
  1. CONSTRAINT `comments_ibfk_1` FOREIGN KEY (`blog_id`) REFERENCES `blogs` (`id`) ON UPDATE CASCADE 

實際上,這個語句是通知MySQLMySQL,當blogs表更新時,也要更新comments表中外鍵blog_id的值。換句話

說,這里所做的就是讓MySQL以級聯方式維護數據庫完整性,這意味著當某個博客更新時,與之相連的注釋也要立即反應此變化,重要的是這一功能的實現并非在應用程序層完成的。

兩個示例MySQL表已經定義好了,現在,更新這兩個表就像運行一個UPDATE語句一樣簡單,如下所示:

 

 
  1. "UPDATE blogs SET id = 2, title = "Title of the first blog entry", content = 'Content of the first blog entry', author = 'John Doe' WHERE id = 1" 

前面說過,我們無需更新comments表,因為MySQL會自動處理這一切。此外,在試圖更新blogs表的數據行的時候,還可以通過去除查詢的“ON UPDATE”部分或者規定“NO ACTION”和“RESTRICT”讓MySQL什么也不做。當然,還可以讓MySQL做其他事情,這些將在后續的文章中分別加以介紹。

通過上面的介紹,我想大家已經對如何在MySQL中的InnoDB表結合使用外鍵約束有了一個清晰的認識,當然,您也可以進一步編寫在即的代碼,以進一步加深對這一方便的數據庫功能的認識。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲一区二区久久久久久| www.日韩欧美| 亚洲乱码国产乱码精品精天堂| 国产精品久久久久久久久久久新郎| 久久久久久亚洲精品不卡| 欧美中文在线观看国产| 欧美视频免费在线| 亚洲美女又黄又爽在线观看| 成人免费看黄网站| 1769国内精品视频在线播放| 777午夜精品福利在线观看| 少妇久久久久久| 亚洲人成网站在线播| 中文字幕国产精品| 久久精品成人欧美大片| 久久久久久久一区二区三区| 亚洲最大av网| 亚洲成人网久久久| 国产精品久久色| 国产亚洲欧洲高清一区| 中文字幕一区电影| 欧美一区二区大胆人体摄影专业网站| 国产成人免费av电影| 久久精品国产亚洲7777| 国产精品九九久久久久久久| 亚洲图片欧美午夜| 日韩在线观看免费全集电视剧网站| 日本欧美精品在线| 国产精品美女久久久久av超清| 国产啪精品视频| 欧美日韩国产影院| 91成人免费观看网站| 亚洲人成网站777色婷婷| 久久久电影免费观看完整版| 国产www精品| 亚洲人高潮女人毛茸茸| 国产精品日日做人人爱| 国产色婷婷国产综合在线理论片a| 国产91精品在线播放| 国产精品成人免费视频| 欧美日韩中文在线观看| 最新国产精品亚洲| 精品日韩视频在线观看| 日韩激情av在线免费观看| 日韩网站免费观看| 久久久日本电影| 91免费视频网站| 欧美国产日韩二区| 亚洲在线视频观看| 国语自产精品视频在线看抢先版图片| 欧美国产视频日韩| 国产精品久久久久久av福利| 欧美精品aaa| 欧美日韩中文字幕在线视频| 亚洲精品女av网站| 亚洲福利精品在线| 7777精品视频| 黄色一区二区在线观看| 亚洲国语精品自产拍在线观看| 国产精品国产三级国产专播精品人| 日韩亚洲在线观看| 日韩高清免费观看| 成人免费大片黄在线播放| 日韩大陆毛片av| 上原亚衣av一区二区三区| 欧美中文在线观看| 日韩精品中文字| 欧美视频在线观看 亚洲欧| 色综合91久久精品中文字幕| 91在线|亚洲| 国产精品美女久久久久久免费| 亚洲午夜国产成人av电影男同| 亚洲精品成人久久久| www国产91| 久久久久久网址| 国产日韩视频在线观看| 国产视频久久久| 日韩亚洲第一页| 正在播放欧美视频| 欧美日韩中文字幕综合视频| 欧美成人第一页| 亚洲第一精品福利| 国产一区欧美二区三区| 欧美精品videosex牲欧美| 精品女厕一区二区三区| 国产精品成人一区| 欧美在线www| 不卡伊人av在线播放| 伊人伊人伊人久久| 国产精品第100页| 欧美激情欧美激情在线五月| 国产精品青青在线观看爽香蕉| 国产精品一区二区三区毛片淫片| 91av网站在线播放| 亚洲色图激情小说| 91精品国产综合久久香蕉922| 亚洲精品电影在线观看| 久久久久久久久久久91| 日韩成人中文字幕在线观看| 最近2019中文字幕第三页视频| 91在线高清免费观看| 欧美精品福利在线| 91精品国产自产在线观看永久| 久久夜色精品国产| 日韩精品在线观看视频| 久久伊人精品视频| 91中文字幕在线| 在线观看久久久久久| 国产精品成人aaaaa网站| 久久91亚洲精品中文字幕| 亚洲经典中文字幕| 日韩三级成人av网| 中日韩午夜理伦电影免费| www.久久撸.com| 亚洲剧情一区二区| 中文字幕九色91在线| 国产三级精品网站| 国产专区欧美专区| 日韩精品中文字| 国产成人在线视频| 日本三级韩国三级久久| 欧美精品xxx| 精品国产欧美成人夜夜嗨| 77777少妇光屁股久久一区| 日韩有码在线电影| 日韩成人高清在线| 久久99精品久久久久久琪琪| 91av在线看| 91精品国产综合久久香蕉最新版| 俺也去精品视频在线观看| 欧美性xxxxx极品娇小| 欧美日韩不卡合集视频| 日本精品久久久久影院| 欧美一区二区三区免费观看| 国产一区二区精品丝袜| 久久手机免费视频| 色偷偷亚洲男人天堂| 欧美一级在线亚洲天堂| 一级做a爰片久久毛片美女图片| 日本aⅴ大伊香蕉精品视频| 日韩电影中文字幕在线| 欧美亚洲成人xxx| 91成人天堂久久成人| 日韩av一区在线观看| 亚洲区一区二区| 欧美激情视频一区二区| 精品亚洲一区二区三区四区五区| 亚洲男人天堂2019| 91在线高清免费观看| 神马国产精品影院av| 亚洲午夜精品久久久久久久久久久久| 亚洲国产精品久久久久秋霞不卡| 国产在线98福利播放视频| 日韩激情av在线播放| 国内精品一区二区三区| 国产成人精品免高潮在线观看| 88xx成人精品| 国内成人精品视频| 国产成人综合av| 亚洲毛片在线观看| 97福利一区二区| 国产午夜精品免费一区二区三区| 欧美日韩一区二区在线播放| 久久精品一本久久99精品|