日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問(wèn)題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷(xiāo)解決方案
MySQL數(shù)據(jù)庫(kù)中的外鍵約束詳解

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

成都創(chuàng)新互聯(lián)-專(zhuān)業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性價(jià)比靜安網(wǎng)站開(kāi)發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫(kù),直接使用。一站式靜安網(wǎng)站制作公司更省心,省錢(qián),快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋靜安地區(qū)。費(fèi)用合理售后完善,10余年實(shí)體公司更值得信賴。

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

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

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

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

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

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

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

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

定義數(shù)據(jù)庫(kù)模式時(shí),我們要在這兩個(gè)表之間建立起一對(duì)多的關(guān)系,方法是在存放評(píng)論的表中創(chuàng)建一個(gè)外鍵,以將其中的數(shù)據(jù)行(即評(píng)論)對(duì)應(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ù)層。如您所見(jiàn),***個(gè)表名為blogs,它由一些含義很明顯的字段組成,分別用于存放每篇博客文章的ID、標(biāo)題和內(nèi)容,***是作者。第二個(gè)表名為comments,用于存放各篇博客文章的有關(guān)評(píng)論,它將博客文章的ID作為它的外鍵,從而建立起一對(duì)多的關(guān)系。

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

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

前面部分,我們創(chuàng)建了兩個(gè)MyISAM表,來(lái)充當(dāng)博客應(yīng)用程序的數(shù)據(jù)層。當(dāng)然,上面的介紹還很簡(jiǎn)單,我們需要做進(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對(duì)我們的***篇博客作出了評(píng)論的情況。假設(shè)現(xiàn)在我們要用另一篇文章來(lái)更新***篇博客。當(dāng)然,這種情況是有可能發(fā)生的。

在這種情況下,為了維護(hù)數(shù)據(jù)庫(kù)的一致性,comments表也必須進(jìn)行相應(yīng)的更新,要么通過(guò)手工方式更新,或者通過(guò)處理數(shù)據(jù)層的應(yīng)用程序進(jìn)行更新。就本例而言,我們將使用SQL命令來(lái)完成更新,具體如下所示:

 
 
 
  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 

如前所述,因?yàn)?**篇博客的數(shù)據(jù)項(xiàng)的內(nèi)容已經(jīng)更新,所以comments表也必須反映出此變化才行。當(dāng)然,現(xiàn)實(shí)中這個(gè)更新操作應(yīng)該在應(yīng)用程序?qū)油瓿?,而非手工進(jìn)行,這就意味著這個(gè)邏輯必須使用服務(wù)器端語(yǔ)言來(lái)實(shí)現(xiàn)。

為了完成這個(gè)操作,對(duì)于PHP來(lái)說(shuō)可以通過(guò)一個(gè)簡(jiǎn)單的子過(guò)程即可,但是實(shí)際上,如果使用了外鍵約束的話,對(duì)comments表的更新操作完全可以委托給數(shù)據(jù)庫(kù)。

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

數(shù)據(jù)庫(kù)的級(jí)聯(lián)更新

下面,我們將利用外鍵約束和InnoDB表(而非默認(rèn)的MyISAM類(lèi)型)來(lái)重新構(gòu)建前面的示例代碼。為此,首先要重新定義這兩個(gè)示例表,以便它們可以使用特定的數(shù)據(jù)庫(kù)引擎。為此,可以使用如下所示的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; 

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

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

實(shí)際上,這個(gè)語(yǔ)句是通知MySQLMySQL,當(dāng)blogs表更新時(shí),也要更新comments表中外鍵blog_id的值。換句話說(shuō),這里所做的就是讓MySQL以級(jí)聯(lián)方式維護(hù)數(shù)據(jù)庫(kù)完整性,這意味著當(dāng)某個(gè)博客更新時(shí),與之相連的注釋也要立即反應(yīng)此變化,重要的是這一功能的實(shí)現(xiàn)并非在應(yīng)用程序?qū)油瓿傻摹?/p>

兩個(gè)示例MySQL表已經(jīng)定義好了,現(xiàn)在,更新這兩個(gè)表就像運(yùn)行一個(gè)UPDATE語(yǔ)句一樣簡(jiǎn)單,如下所示:

"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"

前面說(shuō)過(guò),我們無(wú)需更新comments表,因?yàn)镸ySQL會(huì)自動(dòng)處理這一切。此外,在試圖更新blogs表的數(shù)據(jù)行的時(shí)候,還可以通過(guò)去除查詢的“ON UPDATE”部分或者規(guī)定“NO ACTION”和“RESTRICT”讓MySQL什么也不做。當(dāng)然,還可以讓MySQL做其他事情,這些將在后續(xù)的文章中分別加以介紹。

通過(guò)上面的介紹,我想大家已經(jīng)對(duì)如何在MySQL中的InnoDB表結(jié)合使用外鍵約束有了一個(gè)清晰的認(rèn)識(shí),當(dāng)然,您也可以進(jìn)一步編寫(xiě)在即的代碼,以進(jìn)一步加深對(duì)這一方便的數(shù)據(jù)庫(kù)功能的認(rèn)識(shí)。

【編輯推薦】

  1. 如何在MySQL數(shù)據(jù)庫(kù)中定義外鍵
  2. 詳解MySQL中EXPLAIN解釋命令
  3. MySQL左連接、右連接和內(nèi)連接詳解
  4. MySQL全文檢索中Like索引的實(shí)現(xiàn)
  5. 使用調(diào)度和鎖定進(jìn)行MySQL查詢優(yōu)化

本文名稱(chēng):MySQL數(shù)據(jù)庫(kù)中的外鍵約束詳解
文章地址:http://www.5511xx.com/article/dpcgjeh.html