MySQL表的约束
文章目录
1. 空属性
- 两个值:null(默认的)和not null(不为空)
- 数据库默认字段基本都是字段为空,但是实际开发时,尽可能保证字段不为空,因为数据为空没办法参与运算。
mysql> select null;
+------+
| NULL |
+------+
| NULL |
+------+
1 row in set (0.00 sec)
mysql> select null+1;
+--------+
| null+1 |
+--------+
| NULL |
+--------+
1 row in set (0.00 sec)
案例:
创建一个班级表,包含班级名和班级所在的教室。
站在正常的业务逻辑中:
- 如果班级没有名字,你不知道你在哪个班级
- 如果教室名字可以为空,就不知道在哪上课
所以我们在设计数据库表的时候,一定要在表中进行限制,满足上面条件的数据就不能插入到表中。这就是“约束”。
mysql> create table myclass(class_name varchar(20) not null, class_room varchar(20) not null);
Query OK, 0 rows affected (0.01 sec)
mysql> desc myclass;
+------------+-------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+------------+-------------+------+-----+---------+-------+
| class_name | varchar(20) | NO | | NULL | |
| class_room | varchar(20) | NO | | NULL | |
+------------+-------------+------+-----+---------+-------+
2 rows in set (0.00 sec)
mysql> insert into myclass values('class1', '101'); # 成功插入
Query OK, 1 row affected (0.01 sec)
mysql> insert into myclass (class_name)values('class2'); # 部分插入,剩余使用默认值
ERROR 1364 (HY000): Field 'class_room' doesnot have a default value
mysql> insert into myclass values('class2', null); # 向添加了非空约束的字段传入 null 值
ERROR 1048 (23000): Column 'class_room' cannot be null
2. 默认值
默认值:某一种数据会经常性的出现某个具体的值,可以在一开始就指定好,在需要真实数据的时候,用户可以选择性的使用默认值。
mysql> create table myclass(
-> class_name varchar(20) not null,
-> class_room varchar(20) default '105');
Query OK, 0 rows affected (0.02 sec)
mysql> insert into myclass (class_name)values('class1');
Query OK, 1 row affected (0.01 sec)
mysql> select * from myclass;
+------------+------------+
| class_name | class_room |
+------------+------------+
| class1 | 105 |
+------------+------------+
1 row in set (0.00 sec)
通过上面案例我们发现,当数据在插入的时候不给该字段赋值时,就使用默认值。
注意:只有设置了default的列,才可以在插入值的时候,对列进行省略
3. 列描述 comment
列描述:comment
,没有实际含义,专门用来描述字段,会根据表创建语句保存,用来给程序员或DBA来进行了解。
mysql> create table person(name varchar(20) not null comment '姓名', sex enum('男', '女') default '男' comment '性别', age tinyint unsigned default 0 comment '年龄');
Query OK, 0 rows affected (0.04 sec)
# 注意:not null和defalut一般不需要同时出现,因为default本身有默认值,不会为空
# 使用 desc 查不到comment列描述内容
mysql> desc person;
+-------+-------------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+-------+-------------------+------+-----+---------+-------+
| name | varchar(20) | NO | | NULL | |
| sex | enum('男','女') | YES | | 男 | |
| age | tinyint unsigned | YES | | 0 | |
+-------+-------------------+------+-----+---------+-------+
3 rows in set (0.00 sec)
# 使用 show create table 可以查询
mysql> show create table person\G;
*************************** 1. row ***************************
Table: person
Create Table: CREATE TABLE `person` (
`name` varchar(20) NOT NULL COMMENT '姓名',
`sex` enum('男','女') DEFAULT '男' COMMENT '性别',
`age` tinyint unsigned DEFAULT '0' COMMENT '年龄'
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci
1 row in set (0.00 sec)
4. zerofill
首先利用create语句创建一个含有数字类型字段的表:
mysql> create table t1(a int default 0, b int unsigned default 0);
Query OK, 0 rows affected (0.01 sec)
刚开始学习数据库时,对数字类型后面的长度很迷茫。通过show看看t1表的建表语句:
mysql> show create table t1\G;
*************************** 1. row ***************************
Table: t1
Create Table: CREATE TABLE `t1` (
`a` int DEFAULT '0',
`b` int unsigned DEFAULT '0'
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci
1 row in set (0.00 sec)
ERROR:
No query specified
接着插入一组值,并执行 select 语句打印:
mysql> insert into t1 values(1, 2);
Query OK, 1 row affected (0.00 sec)
mysql> select * from t1;
+------+------+
| a | b |
+------+------+
| 1 | 2 |
+------+------+
1 row in set (0.00 sec)
我们保持字段 a 的数据类型不变,添加 zerofill
约束:
mysql> alter table t1 change a a int zerofill;
Query OK, 1 row affected, 1 warning (0.05 sec)
Records: 1 Duplicates: 0 Warnings: 1
mysql> select * from t1;
+------------+------+
| a | b |
+------------+------+
| 0000000001 | 2 |
+------------+------+
1 row in set (0.00 sec)
这次可以看到a的值由原来的1变成0000000001,这就是 zerofill 属性的作用,如果宽度小于设定的宽度(这里设置的是10),自动填充0。要注意的是,这只是最后显示的结果,在MySQL中实际存储的还是1。数据库内部存储的0000000001只是设置了zerofill属性后的一种格式化输出而已。
5. 主键
主键:primary key用来唯一的约束该字段里面的数据,不能重复,不能为空,一张表中最多只能有一个主键;主键所在的列通常是整数类型。
案例:
- 创建表的时候直接在字段上指定主键
mysql> create table student(
-> id int unsigned primary key comment '学号不能为空',
-> name varchar(20) not null);
Query OK, 0 rows affected (0.01 sec)
mysql> desc student;
+-------+--------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+-------+--------------+------+-----+---------+-------+
| id | int unsigned | NO | PRI | NULL | |
| name | varchar(20) | NO | | NULL | |
+-------+--------------+------+-----+---------+-------+
2 rows in set (0.00 sec)
- 主键约束:主键对应的字段中不能重复,一旦重复,操作失败。
mysql> insert into student values(1, '张三');
Query OK, 1 row affected (0.00 sec)
mysql> insert into student values(1, '李四');
ERROR 1062 (23000): Duplicate entry '1' for key 'student.PRIMARY'
- 当表创建好以后但是没有主键的时候,可以再次追加主键
alter table 表名 add primary key(字段列表)
mysql> alter table student add primary key(id);
Query OK, 0 rows affected (0.02 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> desc student;
+-------+--------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+-------+--------------+------+-----+---------+-------+
| id | int unsigned | NO | PRI | NULL | |
| name | varchar(20) | NO | | NULL | |
+-------+--------------+------+-----+---------+-------+
2 rows in set (0.00 sec)
- 删除主键
alter table 表名 drop primary key;
mysql> alter table student drop primary key;
Query OK, 1 row affected (0.01 sec)
Records: 1 Duplicates: 0 Warnings: 0
mysql> desc student;
+-------+--------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+-------+--------------+------+-----+---------+-------+
| id | int unsigned | NO | | NULL | |
| name | varchar(20) | NO | | NULL | |
+-------+--------------+------+-----+---------+-------+
2 rows in set (0.00 sec)
- 复合主键
在创建表的时候,在所有字段之后,使用primary key(主键字段列表)来创建主键,如果有多个字段作为主键,可以使用复合主键。
mysql> create table course(
-> id int unsigned not null,
-> name varchar(20) not null,
-> student_num int unsigned not null default 0,
-> primary key(id, name));
Query OK, 0 rows affected (0.04 sec)
mysql> desc course;
+-------------+--------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+-------------+--------------+------+-----+---------+-------+
| id | int unsigned | NO | PRI | NULL | |
| name | varchar(20) | NO | PRI | NULL | |
| student_num | int unsigned | NO | | 0 | |
+-------------+--------------+------+-----+---------+-------+
3 rows in set (0.00 sec)
一个表只能有一个主键,但是一个主键可以由多个字段构成!
检验复合主键的冲突情况:
mysql> insert into course values(1, '语文', 103);
Query OK, 1 row affected (0.00 sec)
mysql> insert into course values(2, '语文', 103);
Query OK, 1 row affected (0.00 sec)
mysql> insert into course values(1, '数学', 103);
Query OK, 1 row affected (0.00 sec)
mysql> insert into course values(1, '语文', 105);
ERROR 1062 (23000): Duplicate entry '1-语文' for key 'course.PRIMARY'
mysql> select * from course;
+----+--------+-------------+
| id | name | student_num |
+----+--------+-------------+
| 1 | 数学 | 103 |
| 1 | 语文 | 103 |
| 2 | 语文 | 103 |
+----+--------+-------------+
3 rows in set (0.00 sec)
发现只有当设置为主键的所有字段同时发生冲突时,才会发生主键冲突。
6. 自增长 auto_increment
auto_increment
:当对应的字段,不给值,会自动的被系统触发,系统会从当前字段中已经有的最大值
+1操作,得到一个新的不同的值。通常和主键搭配使用,作为逻辑主键。
自增长的特点:
- 任何一个字段要做自增长,前提是本身是一个索引(key一栏有值)
- 自增长字段必须是整数
- 一张表最多只能有一个自增长
案例:
mysql> create table student(
-> id int unsigned primary key auto_increment,
-> name varchar(10) not null);
Query OK, 0 rows affected (0.01 sec)
mysql> insert into student (name)values('张三');
Query OK, 1 row affected (0.00 sec)
mysql> insert into student (name)values('李四');
Query OK, 1 row affected (0.01 sec)
mysql> insert into student (id, name)values(5, '王五');
Query OK, 1 row affected (0.00 sec)
mysql> select * from student;
+----+--------+
| id | name |
+----+--------+
| 1 | 张三 |
| 2 | 李四 |
| 5 | 王五 |
+----+--------+
3 rows in set (0.00 sec)
在插入后获取上次插入的 AUTO_INCREMENT 的值(批量插入获取的是第一个值)
mysql> select last_insert_id();
+------------------+
| last_insert_id() |
+------------------+
| 2 |
+------------------+
1 row in set (0.00 sec)
利用 show create table 语句,查看当前自增到达的值:
mysql> show create table student\G;
*************************** 1. row ***************************
Table: student
Create Table: CREATE TABLE `student` (
`id` int unsigned NOT NULL AUTO_INCREMENT,
`name` varchar(10) NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci
1 row in set (0.00 sec)
索引:
在关系数据库中,索引是一种单独的、物理的对数据库表中一列或多列的值进行排序的一种存储结构,它是某个表中一列或若干列值的集合和相应的指向表中物理标识这些值的数据页的逻辑指针清单。索引的作用相当于图书的目录,可以根据目录中的页码快速找到所需的内容。
索引提供指向存储在表的指定列中的数据值的指针,然后根据您指定的排序顺序对这些指针排序。数据库使用索引以找到特定值,然后顺指针找到包含该值的行。这样可以使对应于表的SQL语句执行得更快,可快速访问数据库表中的特定信息。
7. 唯一键
一张表中有往往有很多字段需要唯一性,数据不能重复,但是一张表中只能有一个主键:唯一键就可以解决表中有多个字段需要唯一性约束的问题。
唯一键的本质和主键差不多,唯一键允许为空,而且可以多个为空,空字段不做唯一性比较。
关于唯一键和主键的区别:
我们可以简单理解成,主键更多的是标识唯一性的。而唯一键更多的是保证在业务上,不要和别的信息出现重复。
场景假设:
比如在公司,我们需要一个员工管理系统,系统中有一个员工表,员工表中有两列信息,一个身份证号码,一个是员工工号,我们可以选择身份号码作为主键。
而我们设计员工工号的时候,需要一种约束:而所有的员工工号都不能重复。
具体指的是在公司的业务上不能重复,我们设计表的时候,需要这个约束,那么就可以将员工工号设计成为唯一键。
一般而言,我们建议将主键设计成为和当前业务无关的字段,这样,当业务调整的时候,我们可以尽量不会对主键做过大的调整。
mysql> create table employee(
-> id char(20) primary key comment '身份证号',
-> workid int unsigned unique comment '工号');
Query OK, 0 rows affected (0.04 sec)
mysql> insert into employee values('618899', 1);
Query OK, 1 row affected (0.00 sec)
mysql> insert into employee values('618898', null);
Query OK, 1 row affected (0.00 sec)
mysql> insert into employee values('618897', null); # 唯一键不对 null 进行唯一性检查
Query OK, 1 row affected (0.01 sec)
mysql> insert into employee values('618896', 1); # 唯一键冲突
ERROR 1062 (23000): Duplicate entry '1' for key 'employee.workid'
我们发现:
- 唯一键不对 null 进行唯一性检查
- 唯一键允许赋值为空
8. 外键
外键用于定义主表和从表之间的关系:外键约束主要定义在从表上,主表则必须是有主键约束或unique约束
当定义外键后,要求外键列数据必须在主表的主键列存在或为null。
语法:
foreign key (字段名) references 主表(列)
案例:
如果将班级表中的数据都设计在每个学生表的后面,那就会出现数据冗余,所以我们只需设计成让 stu->class_id 和 myclass->id 两者之间实现关联关系,即形成外键约束。
对上面示意图进行代码设计:
- 先创建主键表
mysql> create table myclass (
-> id int primary key,
-> name varchar(30) not null comment'班级名'
-> );
Query OK, 0 rows affected (0.03 sec)
- 再创建从表
mysql> create table stu (
-> id int primary key,
-> name varchar(30) not null comment '学生名',
-> class_id int,
-> foreign key (class_id) references myclass(id)
-> );
Query OK, 0 rows affected (0.01 sec)
- 正常插入数据
mysql> insert into myclass values(10, 'C++大牛班'),(20, 'Java大神班');
Query OK, 2 rows affected (0.00 sec)
Records: 2 Duplicates: 0 Warnings: 0
- 插入一个班级号为30的学生,因为没有这个班级,所以插入不成功
mysql> insert into stu values(1, '张三', 30);
ERROR 1452 (23000): Cannot add or update a child row: a foreign key constraint fails (`test`.`stu`, CONSTRAINT `stu_ibfk_1` FOREIGN KEY (`class_id`) REFERENCES `myclass` (`id`))
- 插入班级id为null,比如来了一个学生,目前还没有分配班级
mysql> insert into stu values(2, '李四', null);
Query OK, 1 row affected (0.00 sec)
mysql> select * from stu;
+----+--------+----------+
| id | name | class_id |
+----+--------+----------+
| 2 | 李四 | NULL |
+----+--------+----------+
1 row in set (0.00 sec)
如何理解外键约束:
现实中此时两张表在业务上是有相关性的,但是在业务上没有建立约束关系,那么就可能出现问题。
解决方案就是通过外键完成的。建立外键的本质其实就是把相关性交给mysql去审核了,提前告诉mysql表之间的约束关系,那么当用户插入不符合业务逻辑的数据的时候,mysql不允许你插入。
9. 综合案例 - 商店(商品与客户)
有一个商店的数据,记录客户及购物情况,有以下三个表组成:
- 商品goods (商品编号goods_id, 商品名goods_name, 单价unitprice, 商品类别category, 供应商provider)
- 客户customer (客户号customer_id, 姓名name, 住址address, 邮箱email, 性别sex)
- 购买purchase (购买订单号order_id, 客户号customer_id, 商品号goods_id, 购买数量nums)
要求:
- 每个表的主外键
- 客户的姓名不能为空值
- 邮箱不能重复
- 客户的性别(男,女)
# 先创建主表 goods,customer
mysql> create table goods(
-> goods_id int unsigned primary key,
-> goods_name varchar(20) not null,
-> unit_price int unsigned not null comment '单价',
-> category varchar(10) not null comment '商品类别',
-> provider varchar(20) not null comment '供应商');
Query OK, 0 rows affected (0.01 sec)
mysql> create table customer(
-> customer_id int unsigned primary key,
-> name varchar(10) not null,
-> address varchar(20) not null comment '地址',
-> email varchar(20) not null unique comment '邮箱',
-> sex enum('男', '女') default '男' comment '性别');
Query OK, 0 rows affected (0.01 sec)
# 再创建从表 purchase 记录购买信息
mysql> create table purchase(
-> order_id int unsigned primary key comment '购买订单号',
-> customer_id int ^C
mysql> create table purchase(
-> order_id int unsigned primary key comment '购买订单号',
-> customer_id int unsigned not null comment '消费者id',
-> goods_id int unsigned not null comment '商品号',
-> nums int unsigned not null default 1 comment '购买个数',
-> foreign key (customer_id) references customer(customer_id),
-> foreign key (goods_id) references goods(goods_id));
Query OK, 0 rows affected (0.03 sec)
mysql> desc purchase;
+-------------+--------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+-------------+--------------+------+-----+---------+-------+
| order_id | int unsigned | NO | PRI | NULL | |
| customer_id | int unsigned | NO | MUL | NULL | |
| goods_id | int unsigned | NO | MUL | NULL | |
| nums | int unsigned | NO | | 1 | |
+-------------+--------------+------+-----+---------+-------+
4 rows in set (0.00 sec)
始终遵循先创建主表再创建从表的原则,只有这样才能方便的绑定主表与从表之间的外键关系。