表与表之间建立关系

Posted yangyinyin

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了表与表之间建立关系相关的知识,希望对你有一定的参考价值。

外键

前戏之一对多关系

# 定义一张部门员工表
id      name        gender      dep_name        dep_desc
1           jason           male            教学部             教书育人
2           egon            male            外交部             漂泊游荡
3           tank            male            教学部             教书育人
4           kevin           male            教学部             教书育人
5           owen            female      技术部             技术能力有限部门
"""
把所有数据都存放于一张表的弊端
1.组织结构不清晰
2.浪费硬盘空间
3.扩展性极差
"""
# 上述的弊端产生原因类似于把代码全部写在一个py文件中,你应该怎么做?>>>解耦合!将上述一张表拆成员工和部门两张表!

# 分析表数据之间的关系:多个用户对应一个部门,一个部门对应多个用户。禁止一个用户对应多个部门

# 如何查找表与表之间的关系
'''
以员工和部门表为例。查找表关系需要做到换位思考(站在两边去找表关系)
先站在员工表:
    找员工表的多条数据能否对应部门表的一条数据
    翻译:
        多个员工能否属于一个部门
        可以!之后不能直接下结论,还需要站在部门表的角度再确认关系

再站在部门表:
    找部门表的多条数据能否对应员工表的一条数据
    翻译:
        多个部门能否有同一个员工
        不可以!
只有站在两边表的角度都分析过了,才能够下结论
员工表单向多对一部门表
'''

一对多(Foreign Key)

# 如何查找表与表之间的关系
"""
老师与课程表
1.站在老师表的角度:一名老师能否教授多门课程(限制死,不能,一名老师只能教python,不能同时教python和linux)
2.站在课程表的角度:一门课程能否可以被多个老师教,完全可以!
那就是课程表多对一老师表,如何表示这种关系?在课程表中创建一个字段(tea_id)指向老师表的id字段

学生与班级表
1.站在学生表的角度:???
2.站在班级表的角度:???
那就是学生表多对一班级表,如何表示这种关系?在学生表中创建一个字段(class_id)指向班级表的id字段
"""
# foreign key会带来什么样的效果?
# 1、在创建表时,先建被关联的表dep,才能建关联表emp(必选要先建被关联表)
create table dep(
    id int primary key auto_increment,
    dep_name char(10),
    dep_comment char(60)
);

create table emp(
    id int primary key auto_increment,
    name char(16),
    gender enum('male','female') not null default 'male'  # default后面的默认值空格直接书写即可,不用加括号
    dep_id int,
    foreign key(dep_id) references dep(id)
);
# 2、在插入记录时,必须先插被关联的表dep,才能插关联表emp
insert into dep(dep_name,dep_comment) values
('sb教学部','sb辅导学生学习,教授python课程'),
('外交部','老男孩上海校区驻张江形象大使'),
('nb技术部','nb技术能力有限部门');

insert into emp(name,gender,dep_id) values
('alex','male',1),
('egon','male',2),
('lxx','male',1),
('wxx','male',1),
('owen','male',3);
# 3.更新于删除都需要考虑到关联于被关联的关系>>>同步更新与同步删除
修改emp表中的dep_id字段
update emp set dep_id=100 where id=1;
update dep set id=100 where id=1;
delete from dep where id=2;
# 由于外键的绑定关系,不能直接修改或删除,

delete from emp where id>1 and id<5;
delete from dep where id=2;
# 要想删除,必须先把关联的信息删掉才能删掉被关联的信息


# 解决办法:同步更新,同步删除,on update cascade ,on delete cascade
create table dep(
    id int primary key auto_increment,
    dep_name char(10),
    dep_comment char(60)
);

create table emp(
    id int primary key auto_increment,
    name char(16),
    gender enum('male','female') not null default 'male',
    dep_id int,
    foreign key(dep_id) references dep(id)
    on update cascade
    on delete cascade
);
insert into dep(dep_name,dep_comment) values
('sb教学部','sb辅导学生学习,教授python课程'),
('外交部','老男孩上海校区驻张江形象大使'),
('nb技术部','nb技术能力有限部门');

insert into emp(name,gender,dep_id) values
('alex','male',1),
('egon','male',2),
('lxx','male',1),
('wxx','male',1),
('owen','male',3);

update dep set id=100 where id=2;
delete from dep where id=100;
# 这时候更新或删掉其中一个,另外一个关联的也会被一起更新或删除

多对多

# 图书表与作者表之间的关系
"""
仍然站在两张表的角度:
1.站在图书表:一本书可不可以有多个作者,可以!那就是书多对一作者
2.站在作者表:一个作者可不可以写多本书,可以!那就是作者多对一书
双方都能一条数据对应对方多条记录,这种关系就是多对多!
"""
# 先来想如何创建表?图书表需要有一个外键关联作者,作者也需要有一个外键字段关联图书。问题来了,先创建谁都不合适!如何解决?
# 建立第三张表,该表中有一个字段fk左表的id,还有一个字段是fk右表的id
create table author(
    id int primary key auto_increment,
    name char(16)
);

create table book(
    id int primary key auto_increment,
    bname char(16),
    price int
);

insert into author(name) values
('egon'),
('alex'),
('wxx');
insert into book(bname,price) values
('python从入门到放弃',200),
('葵花宝典',100),
('九阴真经',150),
('九阳神功',50);

create table author2book(
    id int primary key auto_increment,
    author_id int,
    book_id int,
    foreign key(author_id) references author(id)
    on update cascade    # 同步更新
    on delete cascade,   # 同步删除
    foreign key(book_id) references book(id)
    on update cascade    # 同步更新
    on delete cascade    # 同步删除
);

insert into author2book(author_id,book_id) values
(1,3),
(1,4),
(2,2),
(2,4),
(3,1),
(3,2),
(3,3),
(3,4);

一对一

客户表和学生表(老男孩的客户与学生之间,报名之前都是客户,只有报了名的才能是学生)

# 左表的一条记录唯一对应右表的一条记录,反之也一样
create table customer(
    id int primary key auto_increment,
    name char(20) not null,
    qq char(10) not null,
    phone char(16) not null
);

create table student(
    id int primary key auto_increment,
    class_name char(20) not null,
    customer_id int unique,   # 该字段一定要是唯一的
    foreign key(customer_id) references customer(id)  # 外键的字段一定要保证unique
    on delete cascade
    on update cascade
);
# 三种外键关系都是用foreign key,区别在于如何使用以及其他条件限制即可做出三种关系

简单去判断对应关系:

书籍和作者
一本书可不可以被多个作者写 可以
一个作者可不可以写多本书 可以
多对多关系

书籍和出版社
一本书可不可以被多个出版社出版 不可以
一个出版社可不可以出版多本书 可以
一对多关系

作者与作者详情表
两个不可以就是
一对一

修改表

mysql中不识别大小写的

语法:
1. 修改表名
      ALTER TABLE 表名 
                          RENAME 新表名;
2. 增加字段
      ALTER TABLE 表名
                          ADD 字段名  数据类型 [完整性约束条件…],
                          ADD 字段名  数据类型 [完整性约束条件…];
      ALTER TABLE 表名
                          ADD 字段名  数据类型 [完整性约束条件…]  FIRST;
      ALTER TABLE 表名
                          ADD 字段名  数据类型 [完整性约束条件…]  AFTER 字段名;                       
3. 删除字段
      ALTER TABLE 表名 
                          DROP 字段名;
4. 修改字段    # modify只能改字段数据类型完整约束,不能改字段名,但是change可以!
      ALTER TABLE 表名 
                          MODIFY  字段名 数据类型 [完整性约束条件…];
      ALTER TABLE 表名 
                          CHANGE 旧字段名 新字段名 旧数据类型 [完整性约束条件…];
      ALTER TABLE 表名 
                          CHANGE 旧字段名 新字段名 新数据类型 [完整性约束条件…];

复制表

#  复制表结构+记录 (key不会复制: 主键、外键和索引)
create table new_service select *from service;

# 只复制表结构
select * from service where 1=2;    #条件为假,查不到任何记录

create table new1_service select * from service where 1=2;

create table t4 like employees;

以上是关于表与表之间建立关系的主要内容,如果未能解决你的问题,请参考以下文章

在oracle数据库中,如何建立表与表之间的关系?

SQL数据库里面怎样设置表与表之间的关联

SQL数据库中表与表之间的关联怎么建立

数据库表与表之间的关系

SQL表与表之间建立外键约束之后,怎么建立连级更新和删除?

数据库表关系(一堆垛,多对多)