MySQL的多种约束

事实上,写入表的数据只有数据类型上的约束,但是没有其他更为严格为约束(例如 48.2kg 中的 48.2 填在体重字段里是符合浮点类型的约束的,但是万一该数据填入到另外一个身高字段里呢?),就有可能导致用户产生错误输入。通过约束,就可以倒逼程序员做正确的输入,保证数据库中的数据是符合预期的、合法的(数据库处的数据检查几乎是检验数据合法的最后一条防线)。

1.空属性

NULL 表示"没有"(默认)和 NOT NULL 代表"不为空",而 ''"" 表示空字符串。在实际开发中,应该尽可能保证字段不为空,因为数据为空的不能参与运算。只要字段的属性设置为 not null 时,就会硬性要求用户的数据项在对应字段处的输入不为空且不能忽略(除非后续设置了默认值,这个后面提及),否则不允许插入该数据项。

那怎么添加呢?在创建表时,在对应不为空的字段使用 NOT NULL 即可,即:CREATE TABLE IF NOT EXISTS table_name(field_name TYPE NOT NULL);,这样就形成了一种"非空约束"。

2.默认值

某一个字段的数据在大部分情况下都是一个值,只有在少数情况下才会发生变动,因此就需要为字段设置一个默认值,一般字段的默认值为 NULL,可以使用 DESC table_name; 查询 Default 所设定的默认值。其设置方法和非空属性设置类似,即:CREATE TABLE IF NOT EXISTS table_name(field_name TYPE DEFAULT default_value);,并且 NO NULLDEFAULT 可以叠加使用,其意义是:不允许数据为 NULL 值。但是,有一种情况是不允许叠加使用的,也就是 NOT NULL DEFAULT NULL

注意:一般使用 DEFAULT 就不会使用 NOT NULL,因为设置了 DEFAULT 一般就不会为空。

3.列描述

列描述是类似注释一样的存在,没有其他实际含义,主要是给程序员和 DBA 数据库管理员做提示使用,使用方法和之前的语法差不多,也是加在字段后面即可,即:CREATE TABLE IF NOT EXISTS table_name(field_name TYPE COMMENT 'comment_text');,这只是一种软性约束,其说明实质就是一种注释(不过需要注意的是,使用 DESC 是看不到注释信息的,但是使用 SHOW CREATE 是可以的)。

4.zerofill

我们之前创建表格的时候,其实故意忽略了一个问题,请看下面操作:

sql 复制代码
# 创建表时的细节
mysql> create table if not exists my_table(number int unsigned not null);
Query OK, 0 rows affected (0.02 sec)

mysql> desc my_table;
+--------+------------------+------+-----+---------+-------+
| Field  | Type             | Null | Key | Default | Extra |
+--------+------------------+------+-----+---------+-------+
| number | int(10) unsigned | NO   |     | NULL    |       |
+--------+------------------+------+-----+---------+-------+
1 row in set (0.00 sec)

mysql> show create table my_table\G
*************************** 1. row ***************************
       Table: my_table
Create Table: CREATE TABLE `my_table` (
  `number` int(10) unsigned NOT NULL
) ENGINE=InnoDB DEFAULT CHARSET=latin1
1 row in set (0.00 sec)

不是说 int4 字节的数据么?这个 int(10)10 又是什么意思?实际上如果 ZEROFILL 没有被设置,这个 10 是没有太大意义的。如果设置了 ZEROFILL,会根据这个数字来填充 0,直到 0 和数据位数达到该数字(也就是类似 C 语言使用 printf() 的数据补零修改宽度的操作),补零的目的只是为了显示上更加美观简洁一些,使表数据在显示上是等宽的。而我们也可以在 int() 里填上数字来规定等宽的长度,如果用户没有设定宽度,则会填上默认的宽度(不同类型的宽度可能不同)。

sql 复制代码
# 修改数据宽度
mysql> insert into my_table (number) values (5);
Query OK, 1 row affected (0.00 sec)

mysql> insert into my_table (number) values (500);
Query OK, 1 row affected (0.00 sec)

mysql> select * from my_table;
+------------+
| number     |
+------------+
| 0000000005 |
| 0000000500 |
+------------+
2 rows in set (0.00 sec)

mysql> alter table my_table modify number int(2) unsigned zerofill;
Query OK, 0 rows affected (0.08 sec)
Records: 0  Duplicates: 0  Warnings: 0

mysql> select * from my_table;
+--------+
| number |
+--------+
|     05 |
|    500 |
+--------+
2 rows in set (0.00 sec)

注意:补零操作并不影响查找,也是一种软规定。

5.主键

PRIMARY KEY 用来约束字段里的数据为唯一的、不可重复、不为空的,一张表最多只有一个主键,主键所在的列通常是整数类型。使用方法和上述约束类似。主键约束使得某一项数据的某一类型的数据只能有一份。根据主键来搜索,绝对只能检索到一条记录。

而使用 ALTER TABLE table_name DROP PRIMARY KEY; 即可去除表内设置的主键。

而修改某一些为主键时,就需要保证该列没有重复的数据,再使用 ALTER TABLE table_name ADD PRIMARY KEY(field_name);

还可以设置复合主键(多个列/字段组合为一个主键),只需要在定义好字段名称和字段类型后,使用 PRIMARY KEY(field_name_1, field_name_2, ...) 即可,然后使用 DESC 查看表的属性就会发现,设置为主键的相关字段在 key 出都会显示 PRI 的提示字样。

sql 复制代码
# 使用复合主键
mysql> create table my_table(
    -> id int unsigned,
    -> course_id int unsigned comment '课程编号',
    -> score tinyint unsigned comment '课程得分',
    -> primary key(id, course_id)
    -> );
    
mysql> desc my_table;
+-----------+---------------------+------+-----+---------+-------+
| Field     | Type                | Null | Key | Default | Extra |
+-----------+---------------------+------+-----+---------+-------+
| id        | int(10) unsigned    | NO   | PRI | NULL    |       |
| course_id | int(10) unsigned    | NO   | PRI | NULL    |       |
| score     | tinyint(3) unsigned | YES  |     | NULL    |       |
+-----------+---------------------+------+-----+---------+-------+
3 rows in set (0.00 sec)

mysql> insert into my_table values(001, 41, 90);
Query OK, 1 row affected (0.00 sec)

mysql> desc my_table;
+-----------+---------------------+------+-----+---------+-------+
| Field     | Type                | Null | Key | Default | Extra |
+-----------+---------------------+------+-----+---------+-------+
| id        | int(10) unsigned    | NO   | PRI | NULL    |       |
| course_id | int(10) unsigned    | NO   | PRI | NULL    |       |
| score     | tinyint(3) unsigned | YES  |     | NULL    |       |
+-----------+---------------------+------+-----+---------+-------+
3 rows in set (0.00 sec)

mysql> insert into my_table values(001, 41, 90);
Query OK, 1 row affected (0.00 sec)

mysql> insert into my_table values(002, 41, 89);
Query OK, 1 row affected (0.00 sec)

mysql> select * from my_table;
+----+-----------+-------+
| id | course_id | score |
+----+-----------+-------+
|  1 |        41 |    90 |
|  2 |        41 |    89 |
+----+-----------+-------+
2 rows in set (0.00 sec)

mysql> insert into my_table values(001, 41, 78);
ERROR 1062 (23000): Duplicate entry '1-41' for key 'PRIMARY'

可以看到,复合主键中的各个字段是允许重复的,但是整体不能重复,要统一看作一个主键。

6.自增长

AUTO_INCREMENT 实际上是主键的一种(一般会先设置为主键约束),一张表最多只有一个自增长约束,且对应字段必须是整型,使用方法和前面类似。

设定完后,插入一个记录就会自动加 1(从 1 开始自增,当然也可以自己插入初始值)。

另外,还可以使用 SELECT LAST_INSERT_ID() 来查询最后一个增长的最大数字。

同样,一个表一般只有一个自增长约束。

7.唯一键

和主键类似,唯一键约束 unique 可以保证一个字段的数据保持唯一性(不会出现重复的数据),但是和主键不同的是,一张表只能有一个主键,但可以有多个唯一键。

成功设置为唯一键时,使用 DESC table_name 则会在 key 处显示 UNI

另外,唯一键的字段可以插入空值(但是主键不可以插入为 NULL),并且不作唯一性比较。

补充:主键和唯一键在很多地方时重叠的,但是一般主键是标识一个记录的唯一性,一般和业务无关(这样业务改变也无需变动主键),但是唯一键更多是为了保证在业务上不出现数据重复的问题。两者是互相补充的,不是互相独立、对立的。

8.外键

外键的语法为 [CONSTRAINT foreign_key_name] FOREIGN KEY (field_name) REFERENCES main_table(main_field_name); 有些时候,表与表之间是会产生关系/依赖的。因此外键和之前所有的键不太一样,是根据其他数据表来产生约束的。

sql 复制代码
# 研究使用外键的场景
mysql> create table student( s_id int unsigned primary key auto_increment,
s_name varchar(20) not null, s_telphone varchar(32) not null, c_id int );
Query OK, 0 rows affected (0.04 sec)

mysql> create table class( c_id int unsigned primary key auto_increment, c_
name varchar(20) not null );
Query OK, 0 rows affected (0.04 sec)

mysql> desc student;
+------------+------------------+------+-----+---------+----------------+
| Field      | Type             | Null | Key | Default | Extra          |
+------------+------------------+------+-----+---------+----------------+
| s_id       | int(10) unsigned | NO   | PRI | NULL    | auto_increment |
| s_name     | varchar(20)      | NO   |     | NULL    |                |
| s_telphone | varchar(32)      | NO   |     | NULL    |                |
| c_id       | int(11)          | YES  |     | NULL    |                |
+------------+------------------+------+-----+---------+----------------+
4 rows in set (0.00 sec)

mysql> desc class;
+--------+------------------+------+-----+---------+----------------+
| Field  | Type             | Null | Key | Default | Extra          |
+--------+------------------+------+-----+---------+----------------+
| c_id   | int(10) unsigned | NO   | PRI | NULL    | auto_increment |
| c_name | varchar(20)      | NO   |     | NULL    |                |
+--------+------------------+------+-----+---------+----------------+
2 rows in set (0.00 sec)

mysql> insert into class (c_id, c_name) values (1, '101');
Query OK, 1 row affected (0.00 sec)

mysql> insert into class (c_name) values ('102');
Query OK, 1 row affected (0.00 sec)

mysql> insert into class (c_name) values ('103');
Query OK, 1 row affected (0.01 sec)

mysql> select * from class;
+------+--------+
| c_id | c_name |
+------+--------+
|    1 | 101    |
|    2 | 102    |
|    3 | 103    |
+------+--------+
3 rows in set (0.01 sec)

mysql> insert into student (s_id, s_name, s_telphone, c_id) values (1, 'lim
ou', '13455657787', 3);
Query OK, 1 row affected (0.00 sec)

mysql> insert into student (s_name, s_telphone, c_id) values ('dimou', '135
55652387', 3);
Query OK, 1 row affected (0.00 sec)

mysql> insert into student (s_name, s_telphone, c_id) values ('eimou', '13255232000', 2);
Query OK, 1 row affected (0.00 sec)

mysql> insert into student (s_name, s_telphone, c_id) values ('wimou', '135
44232001', 1);
Query OK, 1 row affected (0.01 sec)

mysql> select * from student;
+------+--------+-------------+------+
| s_id | s_name | s_telphone  | c_id |
+------+--------+-------------+------+
|    1 | limou  | 13455657787 |    3 |
|    2 | dimou  | 13555652387 |    3 |
|    3 | eimou  | 13255232000 |    2 |
|    4 | wimou  | 13544232001 |    1 |
+------+--------+-------------+------+
4 rows in set (0.00 sec)

上述的学生信息录入的过程确实没毛病,但是程序员很有可能给一个学生插入一个不存在的 c_id(甚至是将原班级删除导致 c_id 不存在),为了防止这种情况下的产生,就诞生了外键。

sql 复制代码
# 使用外键
# 先删除原先的数据表
mysql> drop table student;

# 重新建立带有外键的数据表
mysql> create table student(s_id int unsigned primary key, s_name varchar(20) not null, s_telphone varchar(32) unique, c_id int unsigned, foreign key
(c_id) references class(c_id) );
Query OK, 0 rows affected (0.06 sec)

mysql> desc student;
+------------+------------------+------+-----+---------+-------+
| Field      | Type             | Null | Key | Default | Extra |
+------------+------------------+------+-----+---------+-------+
| s_id       | int(10) unsigned | NO   | PRI | NULL    |       |
| s_name     | varchar(20)      | NO   |     | NULL    |       |
| s_telphone | varchar(32)      | YES  | UNI | NULL    |       |
| c_id       | int(10) unsigned | YES  | MUL | NULL    |       |
+------------+------------------+------+-----+---------+-------+
4 rows in set (0.00 sec)

mysql> select * from class;
+------+--------+
| c_id | c_name |
+------+--------+
|    1 | 101    |
|    2 | 102    |
|    3 | 103    |
+------+--------+
3 rows in set (0.00 sec)

# 尝试插入数据
mysql> insert into student values (1, 'limou', '13455657787', 3);
Query OK, 1 row affected (0.01 sec)

mysql> insert into student values (2, 'dimou', '13555652387', 3);
Query OK, 1 row affected (0.00 sec)

mysql> insert into student values (3, 'eimou', '13255232000', 2);
Query OK, 1 row affected (0.00 sec)

mysql> insert into student values (4, 'wimou', '13544232001', 1);
Query OK, 1 row affected (0.01 sec)

mysql> insert into student values (5, 'fimou', '13514875183', 4);
ERROR 1452 (23000): Cannot add or update a child row: a foreign key constraint fails (`limou_database`.`student`, CONSTRAINT `student_ibfk_1` FOREIGN KEY (`c_id`) REFERENCES `class` (`c_id`))

# 尝试直接删除被子表引用为外键的父行
mysql> select * from student;
+------+--------+-------------+------+
| s_id | s_name | s_telphone  | c_id |
+------+--------+-------------+------+
|    1 | limou  | 13455657787 |    3 |
|    2 | dimou  | 13555652387 |    3 |
|    3 | eimou  | 13255232000 |    2 |
|    4 | wimou  | 13544232001 |    1 |
|    6 | fimou  | 13514875183 |    2 |
|    7 | fimou  | 13514875123 |    3 |
+------+--------+-------------+------+
6 rows in set (0.00 sec)

mysql> select * from class;
+------+--------+
| c_id | c_name |
+------+--------+
|    1 | 101    |
|    2 | 102    |
|    3 | 103    |
+------+--------+
3 rows in set (0.00 sec)

mysql> delete from class where c_id=3;
ERROR 1451 (23000): Cannot delete or update a parent row: a foreign key constraint fails (`limou_database`.`student`, CONSTRAINT `student_ibfk_1` FOREIGN KEY (`c_id`) REFERENCES `class` (`c_id`)) # 很明显删除失败

# 去除子表对父表的引用,再尝试删除
mysql> delete from student where c_id=3;
Query OK, 3 rows affected (0.01 sec)

mysql> delete from class where c_id=3;
Query OK, 1 row affected (0.00 sec) # 很明显删除成功

可以看到,最后一次插入出现了错误,说明外键约束生效。

另外 [CONSTRAINT foreign_key_name] 实际上是给外键起一个别名,是可选做法,不设定则会有默认的名字。

相关推荐
_Shirley1 小时前
鸿蒙设置app更新跳转华为市场
android·华为·kotlin·harmonyos·鸿蒙
Fre丸子_1 小时前
ffmpeg之播放一个yuv视频
ffmpeg·音视频
小林coding2 小时前
阿里云 Java 后端一面,什么难度?
java·后端·mysql·spring·阿里云
hedalei3 小时前
RK3576 Android14编译OTA包提示java.lang.UnsupportedClassVersionError问题
android·android14·rk3576
锋风Fengfeng3 小时前
安卓多渠道apk配置不同签名
android
18号房客3 小时前
高级sql技巧进阶教程
大数据·数据库·数据仓库·sql·mysql·时序数据库·数据库架构
枫_feng3 小时前
AOSP开发环境配置
android·安卓
yinqinggong3 小时前
从源码编译支持FFmpeg的OpenCV
opencv·ffmpeg
叶羽西4 小时前
Android Studio打开一个外部的Android app程序
android·ide·android studio
翔云1234564 小时前
MySQL purged gtid是如何生成和维护的
数据库·mysql