mysql外键约束及表关系(三)

mysql是关系型数据库,关系数据库,是建立在关系模型基础上的数据库,
现实世界中的各种实体,以及实体之间的各种联系,均用关系模型(table)来表示。

  • 关系模型就是指二维表格模型,因而一个关系型数据库就是由二维表及其之间的联系组成的一个数据组织。
  • 实体:就是数据对象 (可以是有型的,也可以是无型的);

E-R 图 (实体关系模型)

image.png

E-R图也称实体-联系图(Entity Relationship Diagram),提供了表示实体类型、属性和联系的方法,用来描述现实世界的概念模型。它是描述现实世界关系概念模型的有效方法。是表示概念关系模型的一种方式。

  • 用“矩形框”表示实体型,矩形框内写明实体名称;
  • 用“椭圆图框”表示实体的属性,并用“实心线段”将其与相应关系的“实体型”连接起来;
  • 用”菱形框“表示实体型之间的联系成因,在菱形框内写明联系名,并用”实心线段“分别与有关实体型连接起来,同时在”实心线段“旁标上联系的类型(1:1,1:n或m:n)。

外键约束FOREING KEY

外键约束FOREIGN KEY,保持数据一致性,完整性实现一对一或一对多关系。
(因为一个表只存一类信息。用外键来做参照,保证数据的一致性,可以减少数据冗余)
外键约束的要求:
数据表的存储引擎只能为InnoDB #此点已无需关注。
外键列和参照列数据类型一致
外键必须关联到键上面去,一般情况是关联到,另一张表的主键;
--用desc a;查看字段key是否有值,必须关联到key有值的字段;

    #例:
    ##表a
    mysql> create table `a`( 
        -> a_id int primary key,
        -> a_name varchar(20) not null
        -> );
    insert into a values(1,'a1'),(2,'a2');
    
    alter table a modify a_id int auto_increment;
    show create table a;  #因为前面已经插入2个值,这里修改为自增长,则从3开始;
    alter table a auto_increment=6;   # 只能调大,不可以调小,一个表里面只有一个auto_increment,一般和主键一起用。
    insert into a(a_name) value('a6');
        
    ##表b
    create table `b`(
    b_id int primary key,
    b_name varchar(20) not null,
    fy_id int not null,
    constraint AB_id foreign key(fy_id) references `a`(a_id) #fy_id只能取a表a_id字段已有的值,不可以取其他的;constraint可写可不写,用来取外键名的,不写则默认取的。删除是通过外键名来删除;
    );
    
    #删除外键
    alter table `b` drop foreign key AB_id;
    
    #增加外键
    mysql> alter table `b` 
        -> add constraint AB_id foreign key(fy_id) references `a`(a_id);
    
    #一个表只存一类信息。

注:
首先,外键引用的那个列在主表中必须是主键列或者唯一列。
所以1:n的肯定把外键建立在n的那张表上。
1:1,一般要看谁是主表,谁是附属表,外键当然建立在附属表中。
n:m的情况,需要建立一个关系表,两个原表和其关系分别是1:n,1:m

一对多关系(一对多的,一个表的记录与另一个表的多条记录连接。)

举例,学校中一个学院可以有很多的学生,而一个学生只属于某一个学院(通常情况下),学院与学生之间的关系就是一对多的关系,通过外键关联来实现这种关系。(1:n的在多的一方n建立外键,指向1的主键)

#例:
##创建学院表
mysql> create table department( 
    -> d_id int primary key AUTO_INCREMENT,
    -> d_name varchar(20) not null
    -> );
    
##创建学生表
mysql> create table student(
    -> s_id int primary key AUTO_INCREMENT,
    -> s_name varchar(20) not null,
    -> dept_id int not null,    # 一个不同列, 关联到另一种表的主键
    -> constraint SD_id foreign key(dept_id) references department(d_id)
    -> );


#插入数据
mysql> INSERT INTO `department`(`d_name`)
    -> VALUES('计算机学院'),
    ->       ('外语学院')
    -> ;
Query OK, 2 rows affected (0.10 sec)
Records: 2  Duplicates: 0  Warnings: 0

mysql> INSERT INTO `student`(`s_name`,`dept_id`)
    -> VALUES('s1',1),
    ->       ('s2',2),
    ->       ('s3',2)
    -> ;
Query OK, 3 rows affected (0.08 sec)
Records: 3  Duplicates: 0  Warnings: 0

一对一关系(一个表的记录只能与另一个表的一条记录连接)

举例,学生表中有学号、姓名、学院,但学生还有些比如电话,家庭住址等比较私密的信息,这些信息不会放在学生表当中,会新建一个学生的详细信息表来存放。这时的学生表和学生的详细信息表两者的关系就是一对一的关系,因为一个学生只有一条详细信息。用外键加主键的方式来实现这种关系。(1:1,一般要看谁是主表,谁是附属表,外键当然建立在附属表中。在附属表增加一个主键关联从表,增加一个主键关联主表主键) 。

#例:
#学生表:
mysql> DESC `student`;
+---------+-------------+------+-----+---------+----------------+
| Field   | Type        | Null | Key | Default | Extra          |
+---------+-------------+------+-----+---------+----------------+
| id      | int(11)     | NO   | PRI | NULL    | auto_increment |
| name    | varchar(20) | NO   |     | NULL    |                |
| dept_id | int(11)     | YES  | MUL | NULL    |                |
+---------+-------------+------+-----+---------+----------------+
3 rows in set (0.06 sec)

#建立详细学生表:
mysql> CREATE TABLE `student_details`(
    -> `id` INT PRIMARY KEY,
    -> `sex` VARCHAR(10) not null,
    -> `age` INT,
    -> `address` VARCHAR(20) comment '家庭住址',
    -> `parents` VARCHAR(20),
    -> `home_num` VARCHAR(20),
    -> FOREIGN KEY (`id`) REFERENCES `student`(`s_id`)
    -> );   # 主键映射 主键
Query OK, 0 rows affected (0.67 sec)

insert into student_details value(2,'男',18,'湖南长沙','佳爸爸','123456789');


insert into student_details value(4,'男',20,'湖南长沙','李爸爸','12541656469');

insert into student_details value(5,'男',20,'湖南长沙','李爸爸','12541656469');

多对多关系

(多对多的关系,是通过中间表。中间表是多的一方,所以多的一方要包含指向一的一方的外键。所以中间表,作为两边多的一方,中间表要包含两边的外键,两个外键分别指向两张表的主键。)
举例,学生要报名选修课,一个学生可以报名多门课程,一个课程有很多的学生报名,那么学生表和课程表两者就形成了多对多关系。对于多对多关系,需要创建第三张关系表,关系表中通过外键加主键的形式实现这种关系。(n:m的情况,需要建立一个关系表,两个原表和其关系分别是1:n,1:m)

#例:
#建立课程表:
mysql> CREATE TABLE `course`(
    -> `id` INT PRIMARY KEY AUTO_INCREMENT,
    -> `name` VARCHAR(20) NOT NULL
    -> );
Query OK, 0 rows affected (1.18 sec)

insert into course(name)  values('python'),('java'),('英语'),('日语');

#学生与课程多对多关系表 (中间表)
mysql> CREATE TABLE `select`(
    -> `s_id` INT,  #用来对应学生id
    -> `crs_id` INT, #用来对应课程id
    # 防止同一个学生,多次选同一门课程
    -> PRIMARY KEY (`s_id`,`crs_id`),
   #  必须学生表的学生,才能选课程表中的课程
    -> FOREIGN KEY (`s_id`) REFERENCES `student` (`s_id`), 
    # 只有课程存在,才能选
    -> FOREIGN KEY (`crs_id`) REFERENCES `course` (`id`)
    -> );
Query OK, 0 rows affected (0.50 sec)

insert into `select` values(1,1),(1,3),(4,4);
insert into `select` values(2,3),(2,1),(2,4);

外键约束的参照操作:

mysql> SELECT * FROM `department`;
+----+----------------------+------+
| d_id | name                 | code |
+-----+----------------------+------+
|  1  | 计算机学院            |    1 |
|  2  | 外国语学院            |    2 |
+-----+----------------------+------+

mysql> SELECT * FROM `student`;
+----+--------+--------+
| id | name   | dep_id |
+----+--------+--------+
|  1 | budong |      1 |
|  2 | Tuple  |      1 |
|  3 | Which  |      2 |
+----+--------+--------+

mysql> DELETE FROM `department` WHERE `d_id`=1;
ERROR 1451 (23000): Cannot delete or update a parent row: a foreign key constraint fails (`mydb`.`student`, CONSTRAINT `stu_dep_for_key` FOREIGN KEY (`dep_id`) REFERENCES `department` (`id`))
#因为外键关联的原因,不能删除父表中的记录

mysql> UPDATE `student` SET `dept_id`=1 WHERE `dept_id`=2;
mysql> DELETE FROM `department` WHERE `d_id`=2;
#因为student表中的dep_id没有值关联department中的id=2的数据,所以可以删除department表中id=2的数据
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容