快速理解MySQL中主键与外键的实例教程
主键与外键的关系,通俗点儿讲,我现在有一个论坛,有两张表,一张是主贴thread,一张是回帖reply
先说说主键,主键是表里面唯一识别记录的字段,一般是帖子id,体现在访问的时候,例如是
thread.php?id=1 表示我要访问的是帖子id是1的帖子~
再来说说外键,当我们删除某个帖子的时候,需要执行另一个操作,就是删除所有回帖,如果正常情况下,我们需要执行两次delete操作(thread和reply),这时候如果存在外键,例如,在reply表里面建立一个指向thread表的主键(id)的外键(这个外键绑的字段,必须是对应帖子的id),并指定响应delete,那你在删除thread的时候,mysql自己会帮你把reply表中这个帖子的回复都删掉,而不需要你手动再去执行一次reply表的delete操作~
至于两者之间的关系,在刚才的例子中,reply表的外键,指向的就是thread表的主键~~
搞个例子,简单演示一下使用,做dage和xiaodi两个表,大哥表是主键,小弟表是外键:
建表:
CREATETABLE`dage`( `id`int(11)NOTNULLauto_increment, `name`varchar(32)default'', PRIMARYKEY(`id`) )ENGINE=InnoDBDEFAULTCHARSET=latin1; CREATETABLE`xiaodi`( `id`int(11)NOTNULLauto_increment, `dage_id`int(11)defaultNULL, `name`varchar(32)default'', PRIMARYKEY(`id`), KEY`dage_id`(`dage_id`), CONSTRAINT`xiaodi_ibfk_1`FOREIGNKEY(`dage_id`)REFERENCES`dage`(`id`) )ENGINE=InnoDBDEFAULTCHARSET=latin1;
插入个大哥:
mysql>insertintodage(name)values('铜锣湾');
QueryOK,1rowaffected(0.01sec)
mysql>select*fromdage;
+----+--------+ |id|name| +----+--------+ |1|铜锣湾| +----+--------+ 1rowinset(0.00sec)
插入个小弟:
mysql>insertintoxiaodi(dage_id,name)values(1,'铜锣湾_小弟A');
QueryOK,1rowaffected(0.02sec)
mysql>select*fromxiaodi;
+----+---------+--------------+ |id|dage_id|name| +----+---------+--------------+ |1|1|铜锣湾_小弟A| +----+---------+--------------+
把大哥删除:
mysql>deletefromdagewhereid=1;
ERROR1451(23000):Cannotdeleteorupdateaparentrow:aforeignkeyconstraintfails(`bstar/xiaodi`,CONSTRAINT`xiaodi_ibfk_1`FOREIGNKEY(`dage_id`)REFERENCES`dage`(`id`))
提示:不行呀,有约束的,大哥下面还有小弟,可不能扔下我们不管呀!
插入一个新的小弟:
mysql>insertintoxiaodi(dage_id,name)values(2,'旺角_小弟A');
2ERROR1452(23000):Cannotaddorupdateachildrow:aforeignkeyconstraintfails(`bstar/xiaodi`,CONSTRAINT`xiaodi_ibfk_1`FOREIGNKEY(`dage_id`)REFERENCES`dage`(`id`))
提示:小子,想造反呀!你还没大哥呢!
把外键约束增加事件触发限制:
mysql>showcreatetablexiaodi;CONSTRAINT`xiaodi_ibfk_1`FOREIGNKEY(`dage_id`)REFERENCES`dage`(`id`)
mysql>altertablexiaodidropforeignkeyxiaodi_ibfk_1;
QueryOK,1rowaffected(0.04sec) Records:1Duplicates:0Warnings:
mysql>altertablexiaodiaddforeignkey(dage_id)referencesdage(id)ondeletecascadeonupdatecascade;
QueryOK,1rowaffected(0.04sec) Records:1Duplicates:0Warnings:0
再次试着把大哥删了:
mysql>deletefromdagewhereid=1;
QueryOK,1rowaffected(0.01sec)
mysql>select*fromdage;
Emptyset(0.01sec)
mysql>select*fromxiaodi;
Emptyset(0.00sec)
得,这回对应的小弟也没了,没办法,谁让你跟我ondeletecascade了呢!
例子说明的应该蛮清楚了吧,其他功能对应手册自己实践吧!:-)