约束 | 描述 | 关键字 |
---|---|---|
非空约束 | 限制该字段的数据不能为null | not null |
唯一约束 | 保证该字段的所有数据都是唯一、不重复的 | unique |
主键约束 | 主键是一行数据的唯一标识,要求非空且唯一 | primary key |
默认约束 | 保存数据时,如果未指定该字段的值,则采用默认值 | default |
检查约束 | 保证字段值满足某一个条件 | check |
外键约束 | 用来让两张表的数据之间建立连接,保证数据的一致性和完整性 | foreing key |
字段名 | 字段含义 | 字段类型 | 约束条件 | 约束关键字 |
---|---|---|---|---|
id | id唯一标识 | int | 主键,并且自动增长 | primar key,auto_increment |
name | 姓名 | varchar(10) | 不为空,并且唯一 | not null,unique |
age | 年龄 | int | 大于0,并且小于等于120 | check |
status | 状态 | char(1) | 如果没有指定该值,默认为1 | default |
gender | 性别 | char(1) | 无 | 无 |
-- 并且自动增长的意思是该字段的数据不用我们写,由数据库根据数据量自动补充该字段
create database if not exists bilibili; #创建数据库bilibili use bilibili; #选择bilibili数据库,即可进行下面的代码演示 drop database if exists bilibili; #代码演示结束后删除bilibili数据库。这个小文件全部代码演示完再删 show databases; #查看是否删除完成
-- 创建案例所需的用户表
create table user(
id int primary key auto_increment comment '主键',
name varchar(10) not null unique comment '姓名',
age int check ( age > 0 && age <= 120 ) comment '年龄',
status char(1) default '1' comment '状态',
gender char(1) comment '性别'
) comment '用户表';
-- 验证,插入数据即可验证
insert into user(name,age,status,gender) values ('Tom',19,'1','男'),('Jack',21,'0','男'),('Amy',20,'1','女');
insert into user(name,age,status,gender) values ('Tom',20,'0','男'); #第二次插入跟第一次相同的name就会报错,因为name是唯一
insert into user(name,age,status,gender) values (null,18,'1','男'); #插入的name为null就会报错
insert into user(name,age,status,gender) values ('lisa',130,'0','男'); #age没有满足条件就会报错
insert into user(name,age,gender) values ('Frank',20,'男'); #不给status赋值,那么该status就会默认为1
-- 查看user表
select * from user;
-- 删除user表
drop table if exists user;
-- 准备两张表
create table dept(
id int auto_increment comment 'ID' primary key,
name varchar(50) not null comment '部门名称'
) comment '部门表';
insert into dept (id,name) values (1,'研发部'),(2,'市场部'),(3,'财务部'),(4,'销售部'),(5,'总经办');
create table tmp(
id int auto_increment comment 'ID' primary key,
name varchar(50) not null comment '姓名',
age int comment '年龄',
job varchar(20) comment '职位',
salary int comment '薪资',
entrydate date comment '入职时间',
managerid int comment '直属领导ID',
dept_id int comment '部门ID' #注意这个dept_id关联的是上面dept表的主键(id)
) comment '员工表';
insert into tmp (id,name,age,job,salary,entrydate,managerid,dept_id) values
(1,'金庸',66,'总裁',20000,'2000-01-01',null,5),(2,'张无忌',20,'项目经理',12500,'2005-12-05',1,1),
(3,'杨瑶',33,'开发',8400,'2000-11-03',2,1),(4,'韦一笑',48,'开发',11000,'2002-02-05',2,1),
(5,'常遇春',43,'开发',10500,'2004-09-07',3,1),(6,'小邵',19,'网络',6600,'2004-10-12',2,1);
-- 添加外键为fk_tmp_dept_id,tmp表的dept_id字段(这个字段同时也是我们设置的外键)去关联dept表的主键id -- 首先学习如何添加外键,我们要在创建表的时候给该表添加外键,如下
alter table tmp add constraint fk_tmp_dept_id foreign key (dept_id) references dept(id);
fk_tmp_dept_id是自定义的外键名
tmp表添加外键之后,这个外键是dept_ip字段,该字段前面的图标多了一个蓝色的钥匙
验证:
去dept表,删除id为1的一行,删除研发部,会发现报错,原因:当字表(tmp)外键关联了父表之后,就不能删除父表
如何删除外键
alter table 表名 drop foreign key 外键名称;
例如删除上面创建的外键
alter table tmp drop foreign key fk_tmp_dept_id;
验证:
去dept表,删除id为1的一行,删除研发部,不会报错
行为 | 说明 |
---|---|
no action | 当在父表中删除/更新对应记录时,首先检查该记录是否有对应外键,如果有则不允许删除/更新,与restrict一致 |
restrict | 当在父表中删除/更新对应记录时,首先检查该记录是否有对应外键,如果有则不允许删除/更新,与no action一致 |
cascade | 当在父表中删除/更新对应记录时,首先检查该记录是否有对应外键,如果有,则也删除/更新外键在子表中的记录 |
set null | 当在父表中删除对应记录时,首先检查该记录是否有对应外键,如果有则设置子表中该外键值为null,前提是该外键允许取null |
set default | 父表有变更时,子表将外键列设置成一个默认的值,注意Innodb引擎是不支持的 |
-- 我们可以在创建外键的时候来指定删除以及更新的行为(行为也叫规则)
-- 语法,例如要设cascade行为
alter table 表名 add constraint 外键名称 foreign key (外键字段) references 主表名 (主表字段名) on update cascade on delete cascade;
-- 首先我们要删除上面创建的外键
alter table tmp drop foreign key fk_tmp_dept_id;
-- 重新创建外键,在创建外键时指定外键删除和更新行为,例如该外键添加cascade行为
alter table tmp add constraint fk_tmp_dept_id foreign key (dept_id) references dept(id) on update cascade on delete cascade;
-- 验证如下 -- 打开父表dept,双击dept表的id字段的第一行,把原先的1改为6,点击上方绿色向上箭头表示确认修改,然后点击tmp表,会 -- 发现外键字段dept_id的第2至6行的数据全部变成了6。即我们修改了父表,子表的相关数据会同步变化 -- 如果我们直接把父表的6研发部删掉,那么子表的第2至6行的数据就会全部消失删掉
-- 我们还可以演示一下set null行为。由于我们上面删过dept表的第一行数据,所以我们把dept表、tmp表删了重新创建一次
alter table tmp drop foreign key fk_tmp_dept_id;
drop table if exists dept;
drop table if exists tmp;
-- 然后再创建dept表,并加上数据,代码在上面的58行 -- 重新给tmp表创建外键来关联dept表,给该外键添加set null行为
alter table tmp add constraint fk_tmp_dept_id foreign key (dept_id) references dept(id) on update set null on delete set null;
-- 验证如下 -- 打开父表的dept,删除父表的第一行数据,点击上方绿色向上箭头表示确认修改,然后点击tmp表,会 -- 发现外键字段dept_id的第2至6行的数据全部变成了null
-- 注意update后面的更新规则 和 delete后面的删除规则不一定要一样,可以不一样