Maison > base de données > tutoriel mysql > le corps du texte

Une analyse approfondie des 6 types de contraintes courants dans MySQL

青灯夜游
Libérer: 2021-09-16 19:55:03
avant
3106 Les gens l'ont consulté

Une analyse approfondie des 6 types de contraintes courants dans MySQL

Le sens littéral des contraintes est de stipuler ou de limiter la manière dont quelque chose doit être fait. Dans MySQL, les contraintes consistent à spécifier des règles pour les données dans la table de données, c'est-à-dire à limiter les données pour garantir la fiabilité, par exemple. Les valeurs nulles sont autorisées à apparaître dans une certaine colonne. En pratique, nous rencontrerons les types de contraintes suivants.

  • NON NULL : Assurez-vous que la colonne ne peut pas avoir de valeur NULL
  • NOT NULL : 确保列不能有NULL值
  • CHECK : 确保列中的值满足特定条件
  • UNIQUE : 确保一列中的所有值都不同
  • PRIMARY KEYNOT NULLUNIQUE组合,唯一标识表中的每一行
  • FOREIGN KEY : 外键约束
  • DEFAULT : 如果未指定值,则为列设置默认值

【相关推荐:mysql视频教程

约束

1.NULL

MySQL中通过使用NOT NULL确保列中不会出现Null值,创建表时候格式如下:

mysql> create table user(name varchar(255)not null);
Query OK, 0 rows affected (0.06 sec)
Copier après la connexion

如果试图插入一个null值,则会抛出异常。

mysql> insert user values(null);
ERROR 1048 (23000): Column 'name' cannot be null
Copier après la connexion

或者在现有表上新增NOT NULL约束。

mysql> alter table user modify name varchar(255) not null;
Query OK, 0 rows affected (0.07 sec)
Records: 0  Duplicates: 0  Warnings: 0
Copier après la connexion

删除NOT NULL约束。

mysql> alter table user modify name varchar(255)  null;
Query OK, 0 rows affected (0.09 sec)
Records: 0  Duplicates: 0  Warnings: 0
Copier après la connexion

2.CHECK

如果想在列上定义条件约束,可以使用CHECK,比如下面,强制让年龄字段大于18,小于80,否则将会报错。

mysql> create table user(age int(11) check(age>18 and age <80));
Query OK, 0 rows affected, 1 warning (0.06 sec)
Copier après la connexion

插入测试,可以发现9、81在插入的时候抛出异常。

mysql> insert user values(9);
ERROR 3819 (HY000): Check constraint &#39;user_chk_1&#39; is violated.

mysql> insert user values(19);
Query OK, 1 row affected (0.01 sec)

mysql> insert user values(81);
ERROR 3819 (HY000): Check constraint &#39;user_chk_1&#39; is violated.
mysql>
Copier après la connexion

也可以进行多列约束,如年龄必须大于18,城市必须为中国。

mysql> create table user(age int(11),city varchar(255) ,check(age>18 and city=&#39;中国&#39;));
Query OK, 0 rows affected, 1 warning (0.05 sec)
Copier après la connexion

插入测试。

mysql> insert user values(81,&#39;2&#39;);
ERROR 3819 (HY000): Check constraint &#39;user_chk_1&#39; is violated.
mysql> insert user values(8,&#39;2&#39;);
ERROR 3819 (HY000): Check constraint &#39;user_chk_1&#39; is violated.
mysql> insert user values(20,&#39;2&#39;);
ERROR 3819 (HY000): Check constraint &#39;user_chk_1&#39; is violated.
mysql> insert user values(20,&#39;中国&#39;);
Query OK, 1 row affected (0.01 sec)

mysql> insert user values(20,&#39;中国1&#39;);
ERROR 3819 (HY000): Check constraint &#39;user_chk_1&#39; is violated.
mysql> insert user values(85,&#39;中国&#39;);
Query OK, 1 row affected (0.01 sec)

mysql> insert user values(9,&#39;中国&#39;);
ERROR 3819 (HY000): Check constraint &#39;user_chk_1&#39; is violated.
Copier après la connexion

还可以让列值必须在指定集合中,如性别必须在男、女、未知、人妖集合中。

mysql> create table user(sex varchar(255) check (sex in (&#39;男&#39;,&#39;女&#39;,&#39;未知&#39;,&#39;人妖&#39;)));
Query OK, 0 rows affected (0.05 sec)
Copier après la connexion

插入测试。

mysql> insert user values("男");
Query OK, 1 row affected (0.02 sec)

mysql> insert user values("男男");
ERROR 3819 (HY000): Check constraint &#39;user_chk_1&#39; is violated.
mysql> insert user values("女");
Query OK, 1 row affected (0.01 sec)

mysql> insert user values("人妖");
Query OK, 1 row affected (0.00 sec)
Copier après la connexion

为约束命名并删除约束。

mysql> create table user (age int(11) ,constraint CHK_AGE check(age>18));
Query OK, 0 rows affected, 1 warning (0.05 sec)

mysql> insert user values(5);
ERROR 3819 (HY000): Check constraint &#39;CHK_AGE&#39; is violated.

mysql> alter table user drop check CHK_AGE;
Query OK, 0 rows affected (0.03 sec)
Records: 0  Duplicates: 0  Warnings: 0mysql> insert user values(5);
Query OK, 1 row affected (0.01 sec)
Copier après la connexion

但是,这样的写法你见过吗?

猜猜下面的作用是什么。

这其实是一个case when条件判断,让其仅仅可以插入>=18,或者是在0-10之间的数。

CREATE TABLE `user` (`age` int(11) CHECK 
(((case when (`age` >=18) then 1 
else 
(case when age<10 and age >0 then 1 else 2 end) end) =1)));
Copier après la connexion

3.UNIQUE

UNIQUE约束确保列中的没有重复的值,UNIQUEPRIMARY KEY约束都为一列值的唯一性提供保障,但是UNIQUE每个表可以出现多次,而PRIMARY KEY只能出现一个。

如下面name字段不能重复。

mysql> create table user (name varchar(255),unique(name));
Query OK, 0 rows affected (0.07 sec)
Copier après la connexion

插入测试。

mysql> insert user values("张三");
Query OK, 1 row affected (0.02 sec)

mysql> insert user values("张三");
ERROR 1062 (23000): Duplicate entry &#39;张三&#39; for key &#39;user.name&#39;mysql>
Copier après la connexion

对此约束进行起名,并删除。

mysql> create table user (name varchar(255),constraint name_un unique(name));
Query OK, 0 rows affected (0.07 sec)

mysql> insert user values("张三");
Query OK, 1 row affected (0.02 sec)

mysql> insert user values("张三");
ERROR 1062 (23000): Duplicate entry &#39;张三&#39; for key &#39;user.name_un&#39;
mysql> alter table user drop index name_un;
Query OK, 0 rows affected (0.03 sec)
Records: 0  Duplicates: 0  Warnings: 0

mysql> insert user values("张三");
Query OK, 1 row affected (0.02 sec)
Copier après la connexion

插入后可以用以下语句查看创建语句。

mysql> show create table user;
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------+
| Table | Create Table                                                                                                                                                 |
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------+
| user  | CREATE TABLE `user` (
  `name` varchar(255) DEFAULT NULL,
  UNIQUE KEY `name_un` (`name`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci |
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)
Copier après la connexion

要删除UNIQUE约束,可以使用DROP INDEXALTER TABLE语句:

mysql> DROP INDEX name_un ON user;
Query OK, 0 rows affected (0.03 sec)
Records: 0  Duplicates: 0  Warnings: 0

mysql> show create table user;
+-------+-----------------------------------------------------------------------------------------------------------------------------+
| Table | Create Table                                                                                                                |
+-------+-----------------------------------------------------------------------------------------------------------------------------+
| user  | CREATE TABLE `user` (
  `name` varchar(255) DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci |
+-------+-----------------------------------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)
Copier après la connexion

在现有表上添加。

mysql> alter table user add constraint name_un unique(name);
Query OK, 0 rows affected (0.04 sec)
Records: 0  Duplicates: 0  Warnings: 0
Copier après la connexion

4.PRIMARY KEY

通常每个表中包含一个用于唯一标识每一行的值,这个列就被称为PRIMARY KEY。

mysql> create table user (id int(11) ,age int(11),primary key (id));
Query OK, 0 rows affected, 2 warnings (0.06 sec)

mysql> insert user values(1,2);
Query OK, 1 row affected (0.02 sec)

mysql> insert user values(1,2);
ERROR 1062 (23000): Duplicate entry &#39;1&#39; for key &#39;user.PRIMARY&#39;mysql>
Copier après la connexion

5.FOREIGN KEY

FOREIGN KEY用于约束表中的一个字段必须是另一个表中某个字段所存在的值,但是在另一个表中,这个列不一定是主键,但必须是唯一性索引,否则会创建失败。

比如orders表中的userId必须参考user表中的id,如果插入的userId在user表中不存在,则无法插入。

mysql> create table orders (id int(11) primary key ,userId int(11) ,  FOREIGN KEY (userId) REFERENCES user(id) );
Query OK, 0 rows affected, 2 warnings (0.06 sec)

mysql> insert orders values(1,3);
ERROR 1452 (23000): Cannot add or update a child row: a foreign key constraint fails (`t`.`orders`, CONSTRAINT `orders_ibfk_1` FOREIGN KEY (`userId`) REFERENCES `user` (`id`))

mysql> insert orders values(1,1);
Query OK, 1 row affected (0.01 sec)
Copier après la connexion

但是存在一个问题,如果主表(user)中记录被删除或者更新,那orders中的记录该怎么办?,如下面的例子,可以发现直接报错了。

mysql> update user set id =2 where id =1;

Cannot delete or update a parent row: a foreign key constraint fails (`t`.`orders`, CONSTRAINT `orders_ibfk_1` FOREIGN KEY (`userId`) REFERENCES `user` (`id`)
Copier après la connexion

MySQL提供了几个约束可以帮助我们解决这类问题,比如在user表更新时,orders也相继更新。

  1. RESTRICT:如果子表中有记录,则拒绝更新或删除父表中的记录。

  2. CASCADE:更新或删除父表中的记录时,自动更新或删除子表中的记录。

  3. SET NULL:在更新或删除父表记录时,将子表中字段的值设置为空。

可以发现,默认采用的是RESTRICT,下面来修改一下,让在更新时候也同样更新,在删除时候设置null。

mysql> alter table orders add constraint orders_ibfk_1  FOREIGN KEY (`userId`) REFERENCES `user` (`id`) on update cascade on
delete set null;
Query OK, 0 rows affected (0.12 sec)
Records: 0  Duplicates: 0  Warnings: 0
Copier après la connexion

测试更新

mysql> select * from user;
+----+--------+
| id | name   |
+----+--------+
|  1 | 张三   |
+----+--------+
1 row in set (0.00 sec)

mysql> select * from orders;
Empty set (0.00 sec)

mysql> insert orders values (1,1);
Query OK, 1 row affected (0.01 sec)

mysql> update user set id =2 where id =1;
Query OK, 1 row affected (0.01 sec)
Rows matched: 1  Changed: 1  Warnings: 0

mysql> select * from orders;
+----+--------+
| id | userId |
+----+--------+
|  1 |      2 |
+----+--------+
1 row in set (0.01 sec)
Copier après la connexion

测试删除。

mysql> delete from user where id =2;
Query OK, 1 row affected (0.02 sec)

mysql> select * from orders;
+----+--------+
| id | userId |
+----+--------+
|  1 |   NULL |
+----+--------+
1 row in set (0.00 sec)
Copier après la connexion

6.DEFAULT

DEFAULTCHECK : Assurez-vous que la valeur de la colonne répond à des conditions spécifiques

UNIQUE  : Assurez-vous que toutes les valeurs d'une colonne sont différentes
PRIMARY KEY : Combinaison de NOT NULL et UNIQUE, identifiant unique Chaque ligne du tableau

FOREIGN KEY : Contrainte de clé étrangère

DEFAULT : Si aucune valeur n'est spécifiée , définissez la valeur par défaut de la colonne li>

【Recommandations associées : tutoriel vidéo mysql

Contraintes

1.NULL

Dans MySQL, utilisez < code>NOT NULL Assurez-vous que les valeurs Null n'apparaîtront pas dans les colonnes Lors de la création du tableau, le format est le suivant :

Si vous essayez. pour insérer une valeur nulle, une exception sera levée. rrreee🎜Ou ajoutez une nouvelle contrainte NOT NULL sur une table existante. 🎜rrreee🎜Supprimer la contrainte NOT NULL. 🎜rrreee

2.CHECK

🎜Si vous souhaitez définir des contraintes conditionnelles sur la colonne, vous pouvez utiliser CHECK, comme ce qui suit, pour forcer le champ âge à être supérieur à 18, inférieur à 80, sinon une erreur sera signalée. 🎜rrreee🎜Test d'insertion, vous pouvez constater que 9 et 81 génèrent des exceptions lors de l'insertion. 🎜rrreee🎜Vous pouvez également effectuer des contraintes multi-colonnes, telles que l'âge doit être supérieur à 18 ans et la ville doit être la Chine. 🎜rrreee🎜Insérer le test. 🎜rrreee🎜Vous pouvez également spécifier que la valeur de la colonne doit être dans un ensemble spécifié, par exemple, le sexe doit être dans l'ensemble masculin, féminin, inconnu et transexuelle. 🎜rrreee🎜Insérer le test. 🎜rrreee🎜Nommez les contraintes et supprimez les contraintes. 🎜rrreee🎜Mais l'avez-vous déjà vu écrit comme ça ? 🎜🎜Devinez ce que fait ce qui suit. 🎜🎜Il s'agit en fait d'un cas de jugement conditionnel, qui lui permet d'insérer uniquement >=18, ou un nombre compris entre 0 et 10. 🎜rrreee

3.UNIQUE

🎜La contrainte UNIQUE garantit qu'il n'y a pas de valeurs en double dans la colonne , Les contraintes UNIQUE et PRIMARY KEY fournissent des garanties pour l'unicité d'une valeur de colonne, mais UNIQUE peut apparaître plusieurs fois dans chaque table, tandis que PRIMARY KEY peut apparaître. 🎜🎜Par exemple, le champ de nom ci-dessous ne peut pas être répété. 🎜rrreee🎜Insérer le test. 🎜rrreee🎜Nommez cette contrainte et supprimez-la. 🎜rrreee🎜Après l'insertion, vous pouvez utiliser l'instruction suivante pour afficher l'instruction de création. 🎜rrreee🎜Pour supprimer une contrainte UNIQUE, vous pouvez utiliser l'instruction DROP INDEX ou ALTER TABLE : 🎜rrreee🎜ajouter sur une table existante. 🎜rrreee

4.CLÉ PRIMAIRE

🎜Habituellement, chaque table contient une valeur qui identifie de manière unique chaque ligne, et cette colonne est appelée CLÉ PRIMAIRE. 🎜rrreee

5.FOREIGN KEY

🎜FOREIGN KEY est utilisé pour contraindre un champ de la table à être un autre La valeur qui existe dans un certain champ d'une table, mais dans une autre table, cette colonne n'est pas nécessairement la clé primaire, mais doit être un index unique, sinon la création échouera. 🎜🎜Par exemple, l'identifiant utilisateur dans la table des commandes doit faire référence à l'identifiant dans la table utilisateur. Si l'identifiant utilisateur inséré n'existe pas dans la table utilisateur, il ne peut pas être inséré. 🎜rrreee🎜Mais il y a un problème. Si les enregistrements de la table principale (utilisateur) sont supprimés ou mis à jour, que doivent-ils arriver aux enregistrements dans les commandes ? , comme dans l'exemple ci-dessous, vous pouvez constater qu'une erreur est signalée directement. 🎜rrreee🎜MySQL fournit plusieurs contraintes pour nous aider à résoudre ce type de problème. Par exemple, lorsque la table user est mise à jour, les commandes sont également mises à jour les unes après les autres. 🎜
    🎜🎜RESTRICT : Refuser de mettre à jour ou de supprimer des enregistrements dans la table parent s'il y a des enregistrements dans la table enfant. 🎜🎜🎜CASCADE : Mettez à jour ou supprimez automatiquement les enregistrements de la table enfant lors de la mise à jour ou de la suppression d'enregistrements dans la table parent. 🎜🎜🎜SET NULL : définissez la valeur du champ dans la table enfant sur null lors de la mise à jour ou de la suppression des enregistrements de la table parent. 🎜
🎜Vous pouvez constater que RESTRICT est utilisé par défaut. Modifions-le pour qu'il soit également mis à jour lors de la mise à jour et que null soit défini lors de la suppression. 🎜rrreee🎜Mise à jour du test 🎜rrreee🎜Suppression du test. La contrainte 🎜rrreee

6.DEFAULT

🎜DEFAULT est utilisée pour définir une valeur par défaut pour une colonne If. il n'y a pas Si une valeur est attribuée à un champ, le système insérera automatiquement une valeur par défaut pour ce champ. Aucune affectation signifie que ce champ n'est pas spécifié lors de l'insertion des données. Si une valeur nulle est spécifiée, la valeur nulle sera toujours stockée. à la fin. 🎜rrreee🎜🎜Adresse originale : https://juejin.cn/post/7000352993572814885🎜🎜Auteur : i Listen to the Wind Passing Night🎜🎜🎜Pour plus de connaissances sur la programmation, veuillez visiter : 🎜Vidéo de programmation🎜 ! ! 🎜

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

source:juejin.cn
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal
À propos de nous Clause de non-responsabilité Sitemap
Site Web PHP chinois:Formation PHP en ligne sur le bien-être public,Aidez les apprenants PHP à grandir rapidement!