IT貓撲網(wǎng):您身邊最放心的安全下載站! 最新更新|軟件分類|軟件專題|手機(jī)版|論壇轉(zhuǎn)貼|軟件發(fā)布

您當(dāng)前所在位置: 首頁數(shù)據(jù)庫MYSQL → MySQL數(shù)據(jù)庫中的外鍵約束詳解

MySQL數(shù)據(jù)庫中的外鍵約束詳解

時(shí)間:2015-06-28 00:00:00 來源:IT貓撲網(wǎng) 作者:網(wǎng)管聯(lián)盟 我要評論(0)

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

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

像上面這樣的應(yīng)用程序,不僅維護(hù)兩個(gè)表的完整性是一個(gè)嚴(yán)峻的挑戰(zhàn),而最大的難點(diǎn)在于我們必須在應(yīng)用程序級別來維護(hù)它們的完整性。這是大部分不要求使用事務(wù)的web項(xiàng)目在開發(fā)期間所采取的方法,因?yàn)镸yISAM表可以提供出色的性能。

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

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

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

本文將介紹如何將外鍵約束應(yīng)用于InnoDB表。此外,我們還將使用一個(gè)簡單的基于PHP的MySQL抽象類來創(chuàng)建有關(guān)的示例代碼;當(dāng)然,您也可以使用自己喜歡的其它服務(wù)器端語言。現(xiàn)在,我們開始介紹如何將外鍵約束應(yīng)用于MySQL。

使用外鍵約束的時(shí)機(jī)

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

定義數(shù)據(jù)庫模式時(shí),我們要在這兩個(gè)表之間建立起一對多的關(guān)系,方法是在存放評論的表中創(chuàng)建一個(gè)外鍵,以將其中的數(shù)據(jù)行(即評論)對應(yīng)到特定的博客文章。下面是創(chuàng)建示例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;?

?

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

迄今為止,我們的工作還算輕松,因?yàn)槲覀冎皇莿?chuàng)建了兩個(gè)簡單的MyISAM表。下一步,我們要做的是使用一些記錄來填充這些表,以便進(jìn)一步演示在第一個(gè)表中刪除表項(xiàng)時(shí),應(yīng)該在另一個(gè)表中執(zhí)行那些操作。

更新并維護(hù)數(shù)據(jù)庫的完整性

前面部分,我們創(chuàng)建了兩個(gè)MyISAM表,來充當(dāng)博客應(yīng)用程序的數(shù)據(jù)層。當(dāng)然,上面的介紹還很簡單,我們需要做進(jìn)一步的討論。為此,我們將向這些表中填入一些記錄,方法是使用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')?

?

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

在這種情況下,為了維護(hù)數(shù)據(jù)庫的一致性,comments表也必須進(jìn)行相應(yīng)的更新,要么通過手工方式更新,或者通過處理數(shù)據(jù)層的應(yīng)用程序進(jìn)行更新。就本例而言,我們將使用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關(guān)鍵詞標(biāo)簽:MySQL,數(shù)據(jù)庫,MySQL外

相關(guān)閱讀

文章評論
發(fā)表評論

熱門文章 Xbox Game Pass Xbox Game Pass 10款MySQL數(shù)據(jù)庫客戶端圖形界面管理工具推薦 10款MySQL數(shù)據(jù)庫客戶端圖形界面管理工具推薦 MySQL常用維護(hù)管理工具 MySQL常用維護(hù)管理工具 MySQL數(shù)據(jù)庫啟動(dòng)失敗1067進(jìn)程意外終止的解決辦法總結(jié) MySQL數(shù)據(jù)庫啟動(dòng)失敗1067進(jìn)程意外終止的解決辦法總結(jié)

相關(guān)下載

人氣排行 10款MySQL數(shù)據(jù)庫客戶端圖形界面管理工具推薦 MySQL數(shù)據(jù)庫啟動(dòng)失敗1067進(jìn)程意外終止的解決辦法總結(jié) Mysql 1045錯(cuò)誤解決辦法 MySQL服務(wù)器進(jìn)程CPU占用100%解決辦法 MySQL導(dǎo)出導(dǎo)入命令的用例 MySQL連接字符串的實(shí)際操作步驟匯總 MySQL無法啟動(dòng)、無法停止各種解決方法總結(jié) 三種常用的MySQL建表語句