13_MySQL中的约束(三)

本文涉及的产品
云数据库 RDS MySQL Serverless,0.5-2RCU 50GB
云数据库 RDS MySQL Serverless,价值2615元额度,1个月
简介: 13_MySQL中的约束(三)

5. 自增列:AUTO_INCREMENT


5.1 作用


某个字段的值自增


5.2 关键字


auto_increment


5.3 特点和要求


(1)一个表最多 只能有一个 自增长列

(2)当需要产生唯一标识符或顺序值时,可设置自增长

(3)自增长列约束的列必须是 键列 (主键列或唯一键列)

(4)自增约束的列的数据类型必须是 整数类型

(5)如果自增列指定了 0 和 null ,会在当前最大值的基础上自增;如果自增列手动指定了具体               值,直接赋值为具体值。


5.4 如何指定自增约束


( 1 )建表时


create table 表名称(
字段名 数据类型 primary key auto_increment,
字段名 数据类型 unique key not null,
字段名 数据类型 unique key,
字段名 数据类型 not null default 默认值,
);
create table 表名称(
字段名 数据类型 default 默认值 ,
字段名 数据类型 unique key auto_increment,
字段名 数据类型 not null default 默认值,
primary key(字段名)
);


( 2 )建表后


alter table 表名称 modify 字段名 数据类型 auto_increment;


5.5 如何删除自增约束


alter table 表名称 modify 字段名 数据类型 auto_increment;#给这个字段增加自增约束
alter table 表名称 modify 字段名 数据类型; #去掉auto_increment相当于删除


5.6 MySQL 8.0新特性—自增变量的持久化


在 MySQL 8.0 之前,自增主键 AUTO_INCREMENT 的值如果大于 max(primary key)+1 ,在 MySQL 重启后,会重 置AUTO_INCREMENT=max(primary key)+1 ,这种现象在某些情况下会导致业务主键冲突或者其他难以发现的问题。 下面通过案例来对比不同的版本中自增变量是否持久化。 在MySQL 5.7 版本中,测试步骤如下: 创建的数据表中包含自增主键的id 字段,语句如下:


CREATE TABLE test1(
id INT PRIMARY KEY AUTO_INCREMENT
);


插入 4 个空值,执行如下:


INSERT INTO test1
VALUES(0),(0),(0),(0);


查询数据表 test1 中的数据,结果如下:


9b8b6f98d9b24e107c1ce52fedbcab4a_503ee8731ecd4fe4a6b3af3c4faacd63.png


删除id为4的记录,语句如下:


DELETE FROM test1 WHERE id = 4;
再次插入一个空值,语句如下:
INSERT INTO test1 VALUES(0);


查询此时数据表 test1 中的数据,结果如下:


fff636f58b58338ba8cf394b4d437ce9_2e2dbad1842843fd9370e59bcd47e618.png


从结果可以看出,虽然删除了id为4的记录,但是再次插入空值时,并没有重用被删除的4,而是分配了 5。 删除id为5的记录,结果如下:


DELETE FROM test1 where id=5;


重启数据库 ,重新插入一个空值。


INSERT INTO test1 values(0);


再次查询数据表test1中的数据,结果如下:


91b20cc5b066e0e3dbf0771cb18771bf_bec43e8e07b84ddd96545842b58afe6b.png


从结果可以看出,新插入的 0 值分配的是 4 ,按照重启前的操作逻辑,此处应该分配 6 。出现上述结果的主要原因是自增主键没有持久化。 在MySQL 5.7 系统中,对于自增主键的分配规则,是由 InnoDB 数据字典内部一个 计数器 来决定的,而该计数器只在 内存中维护 ,并不会持久化到磁盘中。当数据库重启时,该计数器会被初始化。

在 MySQL 8.0 版本中,上述测试步骤最后一步的结果如下:


b4c4d8e863e75cde398adf6aafd76594_4dd6e3d4a43f49da836057e81ff3a4bd.png


从结果可以看出,自增变量已经持久化了。


MySQL 8.0将自增主键的计数器持久化到 重做日志中。每次计数器发生改变,都会将其写入重做日志中。如果数据库重启,InnoDB会根据重做日志中的信息来初始化计数器的内存值。


6. FOREIGN KEY 约束


6.1 作用


限定某个表的某个字段的 引用完整性 。

比如:员工表的员工所在部门的选择,必须在部门表能找到对应的部分。


6.2 关键字


FOREIGN KEY


6.3 主表和从表/父表和子表


主表(父表):被引用的表,被参考的表

从表(子表):引用别人的表,参考别人的表

例如:员工表的员工所在部门这个字段的值要参考部门表:部门表是主表,员工表是从表。

例如:学生表、课程表、选课表:选课表的学生和课程要分别参考学生表和课程表,学生表和课程表是主表,选课表是从表。


6.4 特点


(1)从表的外键列,必须引用 / 参考主表的 主键或唯一约束 的列

        为什么?因为被依赖 / 被参考的值必须是唯一的

(2)在创建外键约束时,如果不给外键约束命名, 默认名不是列名,而是自动产生一个外键 (例           如student_ibfk_1;),也可以指定外键约束名。

(3)创建 (CREATE) 表时就指定外键约束的话,先创建主表,再创建从表

(4)删表时, 先删从表 (或先删除外键约束), 再删除主表

(5)当主表的记录被从表参照时,主表的记录将不允许删除,如果要删除数据,需要先删除从表           中依赖该记录的数据,然后才可以删除主表的数据

(6)在 “ 从表 ” 中指定外键约束,并且一个表可以建立多个外键约束

(7)从表的外键列与主表被参照的列名字可以不相同,但是 数据类型必须一样 ,逻辑意义一致。           如果类型不一样,创建子表时,就会出现错误“ERROR 1005 (HY000): Can't create                         table'database.tablename'(errno: 150)”。

        例如:都是表示部门编号,都是 int 类型。

(8) 当创建外键约束时,系统默认会在所在的列上建立对应的普通索引 。但是索引名是外键的约           束名。(根据外键查询效率很高)

(9)删除外键约束后,必须 手动 删除对应的索引


6.5 添加外键约束


(1)建表时


create table 主表名称(
字段1 数据类型 primary key,
字段2 数据类型
);
create table 从表名称(
字段1 数据类型 primary key,
字段2 数据类型,
[CONSTRAINT <外键约束名称>] FOREIGN KEY(从表的某个字段) references 主表名(被参考字段)
);
#(从表的某个字段)的数据类型必须与主表名(被参考字段)的数据类型一致,逻辑意义也一样
#(从表的某个字段)的字段名可以与主表名(被参考字段)的字段名一样,也可以不一样
-- FOREIGN KEY: 在表级指定子表中的列
-- REFERENCES: 标示在父表中的列
create table dept( #主表
did int primary key, #部门编号
dname varchar(50) #部门名称
);
create table emp(#从表
eid int primary key, #员工编号
ename varchar(5), #员工姓名
deptid int, #员工所在的部门
foreign key (deptid) references dept(did) 
#在从表中指定外键约束
#emp表的deptid和和dept表的did的数据类型一致,意义都是表示部门的编号
);


说明:


( 1 )主表 dept 必须先创建成功,然后才能创建 emp 表,指定外键成功。

( 2 )删除表时,先删除从表 emp ,再删除主表 dept

(2)建表后

一般情况下,表与表的关联都是提前设计好了的,因此,会在创建表的时候就把外键约束定义好。不过,如果需要修改表的设计(比如添加新的字段,增加新的关联关系),但没有预先定义外键约束,那么,就要用修改表的方式来补充定义。

格式:


ALTER TABLE 从表名 ADD [CONSTRAINT 约束名] FOREIGN KEY (从表的字段) REFERENCES 
主表名(被引用字段) [on update xx][on delete xx];


举例:


ALTER TABLE emp1
ADD CONSTRAINT emp_dept_id_fk FOREIGN KEY(dept_id) REFERENCES dept(dept_id);


总结:约束关系是针对 双方的

添加了外键约束后,主表的修改和删除数据受约束

添加了外键约束后,从表的添加和修改数据受约束

在从表上建立外键,要求主表必须存在

删除主表时,要求 从表先删除 ,或将从表中外键引用该主表的关系先删除


6.6 约束等级


Cascade方式 :在父表上update/delete记录时,同步update/delete掉子表的匹配记录

Set null方式 :在父表上update/delete记录时,将子表上匹配记录的列设为null,但是要注意子表的外键列不能为not null

No action方式 :如果子表中有匹配的记录,则不允许对父表对应候选键进行update/delete操作

Restrict方式 :同no action, 都是立即检查外键约束

Set default方式 (在可视化工具SQLyog中可能显示空白):父表有变更时,子表将外键列设置成一个默认的值,但Innodb不能识别

如果没有指定等级,就相当于 Restrict 方式。

对于外键约束,最好是采用 : ON UPDATE CASCADE ON DELETE RESTRICT 的方式。


(1)演示 1 : on update cascade on delete set null


create table dept(
did int primary key, #部门编号
dname varchar(50) #部门名称
);
create table emp(
eid int primary key, #员工编号
ename varchar(5), #员工姓名
deptid int, #员工所在的部门
foreign key (deptid) references dept(did) on update cascade on delete set null
#把修改操作设置为级联修改等级,把删除操作设置为set null等级
);
insert into dept values(1001,'教学部');
insert into dept values(1002, '财务部');
insert into dept values(1003, '咨询部');
insert into emp values(1,'张三',1001); #在添加这条记录时,要求部门表有1001部门
insert into emp values(2,'李四',1001);
insert into emp values(3,'王五',1002);
select * from dept;
select * from emp;
#修改主表成功,从表也跟着修改,修改了主表被引用的字段1002为1004,
#从表的引用字段就跟着修改为1004了
update dept set did = 1004 where did = 1002;


bfac5dbb155b9be498c38cca67c58e1a_192191aa21c343a396f8a5039fa801e3.png



#删除主表的记录成功,从表对应的字段的值被修改为null
mysql> delete from dept where did = 1001;


be1e40110b39db6191c4ad11760e9475_879c94fb07ae4784a33b3adc98749835.png


874daadd991007ff1f3c88391eecef02_cebfe3d2969943879b8e91478d4f9eb1.png


6.8 开发场景


问题 1 :如果两个表之间有关系(一对一、一对多),比如:员工表和部门表(一对多),它们之间是否 一定要建外键约束?

答:不是的

问题 2 :建和不建外键约束有什么区别?

答:建外键约束,你的操作(创建表、删除表、添加、修改、删除)会受到限制,从语法层面受到限

制。例如:在员工表中不可能添加一个员工信息,它的部门的值在部门表中找不到。

不建外键约束,你的操作(创建表、删除表、添加、修改、删除)不受限制,要保证数据的 引用完整 性 ,只能依 靠程序员的自觉 ,或者是 在 Java 程序中进行限定 。例如:在员工表中,可以添加一个员工的信息,它的部门指定为一个完全不存在的部门。

问题 3 :那么建和不建外键约束和查询有没有关系?

答:没有

在 MySQL 里,外键约束是有成本的,需要消耗系统资源。对于大并发的 SQL 操作,有可能会不适合。比如大型网站的中央数据库,可能会 因为外键约束的系统开销而变得非常慢 。所以, MySQL 允许你不使用系统自带的外键约束,在 应用层面 完成检查数据一致性的逻辑。也就是说,即使你不用外键约束,也要想办法通过应用层面的附加逻辑,来实现外键约束的功能,确保数据的一致性。


6.9 阿里开发规范


【 强制 】不得使用外键与级联,一切外键概念必须在应用层解决。

说明:(概念解释)学生表中的 student_id 是主键,那么成绩表中的 student_id 则为外键。如果更新学生表中的 student_id ,同时触发成绩表中的 student_id 更新,即为级联更新。外键与级联更新适用于 单机低并发 ,不适合 分布式 、 高并发集群 ;级联更新是强阻塞,存在数据库 更新风暴 的风险;外键影响数据库的 插入速度 。


7. CHECK 约束


7.1 作用


检查某个字段的值是否符号 xx 要求,一般指的是值的范围


7.2 关键字


CHECK


7.3 说明:MySQL 5.7 不支持


MySQL5.7 可以使用 check 约束,但 check 约束对数据验证没有任何作用。添加数据时,没有任何错误或警告但是 MySQL 8.0 中可以使用 check 约束了 。


create table employee(
eid int primary key,
ename varchar(5),
gender char check ('男' or '女')
);


8. DEFAULT约束


8.1 作用


给某个字段 / 某列指定默认值,一旦设置默认值,在插入数据时,如果此字段没有显式赋值,则赋值为默认值。


8.2 关键字


DEFAULT


8.3 如何给字段加默认值


(1)建表时


create table 表名称(
字段名 数据类型 primary key,
字段名 数据类型 unique key not null,
字段名 数据类型 unique key,
字段名 数据类型 not null default 默认值,
);

(2)建表后


alter table 表名称 modify 字段名 数据类型 default 默认值;
#如果这个字段原来有非空约束,你还保留非空约束,那么在加默认值约束时,
#还得保留非空约束,否则非空约束就被删除了
#同理,在给某个字段加非空约束也一样,如果这个字段原来有默认值约束,
#你想保留,也要在modify语句中保留默认值约束,否则就删除了
alter table 表名称 modify 字段名 数据类型 default 默认值 not null;


alter table 表名称 modify 字段名 数据类型 ;#删除默认值约束,也不保留非空约束
alter table 表名称 modify 字段名 数据类型 not null; #删除默认值约束,保留非空约束
相关实践学习
基于CentOS快速搭建LAMP环境
本教程介绍如何搭建LAMP环境,其中LAMP分别代表Linux、Apache、MySQL和PHP。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
相关文章
|
1月前
|
存储 关系型数据库 MySQL
【MySQL】6. 表的约束
【MySQL】6. 表的约束
19 0
|
4月前
|
关系型数据库 MySQL 数据库
MySQL数据库——约束
MySQL数据库——约束
|
2月前
|
SQL 关系型数据库 MySQL
【MySQL 数据库】2、MySQL 的数据控制语言、函数和约束
【MySQL 数据库】2、MySQL 的数据控制语言、函数和约束
33 0
|
1月前
|
SQL 关系型数据库 MySQL
轻松入门MySQL:深入学习数据库表管理,创建、修改、约束、建议与性能优化(3)
轻松入门MySQL:深入学习数据库表管理,创建、修改、约束、建议与性能优化(3)
|
1月前
|
存储 关系型数据库 MySQL
MySQL数据库性能大揭秘:表设计优化的高效策略(优化数据类型、增加冗余字段、拆分表以及使用非空约束)
MySQL数据库性能大揭秘:表设计优化的高效策略(优化数据类型、增加冗余字段、拆分表以及使用非空约束)
|
3天前
|
SQL 关系型数据库 MySQL
MySQL数据库的约束+进阶版新增与查询-2
MySQL数据库的约束+进阶版新增与查询
12 1
|
3天前
|
关系型数据库 MySQL 测试技术
MySQL数据库的约束+进阶版新增与查询-1
MySQL数据库的约束+进阶版新增与查询
14 1
|
13天前
|
关系型数据库 MySQL 数据库
【MySQL】:约束全解析
【MySQL】:约束全解析
24 0
|
14天前
|
NoSQL 关系型数据库 MySQL
【MySQL探索之旅】MySQL数据表的增删查改——约束
【MySQL探索之旅】MySQL数据表的增删查改——约束
|
16天前
|
存储 关系型数据库 MySQL
{MySQL} 数据库约束& 表的关系& 新增&&删除& 修改& 查询
{MySQL} 数据库约束& 表的关系& 新增&&删除& 修改& 查询
20 0

推荐镜像

更多