MySQL中的CHECK约束不起作用

首先我创build了一个表

CREATE TABLE Customer ( SD integer CHECK (SD > 0), Last_Name varchar (30), First_Name varchar(30) ); 

然后在该表中插入值

 INSERT INTO Customer values ('-2','abc','zz'); 

MySQL不显示错误,它接受了值。

MySQL参考手册说:

CHECK子句被parsing,但被所有存储引擎忽略。

试试触发器…

 mysql> delimiter // mysql> CREATE TRIGGER trig_sd_check BEFORE INSERT ON Customer -> FOR EACH ROW -> BEGIN -> IF NEW.SD<0 THEN -> SET NEW.SD=0; -> END IF; -> END -> // mysql> delimiter ; 

希望有所帮助。

不幸的是,MySQL不支持SQL检查约束。 你可以在你的DDL查询中定义它们的兼容性,但是它们只是被忽略。

有一个简单的select

您可以创buildBEFORE INSERTBEFORE UPDATE触发器,这些触发器可能会导致错误,或者在不满足数据需求时将字段设置为默认值。

BEFORE INSERT在MySQL 5.5之后工作的例子

 DELIMITER $$ CREATE TRIGGER `test_before_insert` BEFORE INSERT ON `Test` FOR EACH ROW BEGIN IF CHAR_LENGTH( NEW.ID ) < 4 THEN SIGNAL SQLSTATE '12345' SET MESSAGE_TEXT := 'check constraint on Test.ID failed'; END IF; END$$ DELIMITER ; 

在MySQL 5.5之前,你必须导致一个错误,例如调用一个未定义的过程。

在这两种情况下,这会导致隐式事务回滚。 MySQL不允许在过程和触发器中使用ROLLBACK语句。

如果你不想回滚事务(即使失败的“检查约束”,INSERT / UPDATE也应该通过),你可以使用SET NEW.ID = NULL覆盖这个值,这会将id设置为字段默认值,不会真的有意义的一个ID寿

编辑:删除了stream浪的报价。

关于:=运算符:

=不同, :=运算符永远不会被解释为比较运算符。 这意味着您可以在任何有效的SQL语句(不仅仅是在SET语句中)中使用:=为variables赋值。

https://dev.mysql.com/doc/refman/5.6/en/assignment-operators.html

关于反引号标识符的引用:

标识符引号字符是反引号(“`”)

如果启用了ANSI_QUOTES SQL模式,则也可以在双引号内引用标识符

http://dev.mysql.com/doc/refman/5.6/en/identifiers.html

CHECK约束被MySQL忽略,正如在文档CREATE TABLE中的一个小小的注释中所解释的那样

CHECK子句被parsing,但被所有存储引擎忽略。

CHECK约束似乎没有在MySQL中实现。

看到这个错误报告: https : //bugs.mysql.com/bug.php?id = 3464

正如joanq所提到的,MariaDB现在似乎支持CHECK约束和其他好东西:

“支持CHECK CONSTRAINT( MDEV-7563 )”。

https://mariadb.com/kb/en/mariadb/mariadb-1021-release-notes/

试试set sql_mode = 'STRICT_TRANS_TABLES'SET sql_mode='STRICT_ALL_TABLES'