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 INSERT
和BEFORE 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模式,则也可以在双引号内引用标识符
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'