0

0

mysql SQL执行流程中的数据完整性与约束处理

P粉602998670

P粉602998670

发布时间:2026-01-17 12:41:02

|

705人浏览过

|

来源于php中文网

原创

约束检查分阶段穿插进行:非空约束在值解析后、写入前检查;主键和唯一键冲突在行写入前触发;外键检查由InnoDB实时校验,MyISAM忽略;MySQL不支持延迟约束,所有检查均在语句执行后立即进行。

mysql sql执行流程中的数据完整性与约束处理

SQL执行时约束检查发生在哪个阶段

MySQL在INSERTUPDATEDELETE语句执行过程中,**约束检查不是一次性在最后做,而是分阶段穿插进行**。主键(PRIMARY KEY)和唯一键(UNIQUE)冲突检查通常在“行写入前”触发;外键(FOREIGN KEY)检查则依赖存储引擎——InnoDB会在语句执行期间实时校验,而MyISAM直接忽略外键约束。

常见错误现象:ERROR 1062 (23000): Duplicate entry 'xxx' for key 'PRIMARY'ERROR 1452 (23000): Cannot add or update a child row,都说明约束已在语句执行中途被拒绝,事务会回滚到语句级保存点(如果启用了innodb_locks_unsafe_for_binlog等特殊配置则可能不同)。

  • 非空约束(NOT NULL)检查在解析完值、准备写入前发生,早于主键/唯一检查
  • 检查顺序受列定义顺序影响不大,但受索引结构影响:联合唯一索引要求整组值同时满足唯一性
  • SET sql_mode = 'STRICT_TRANS_TABLES'能确保所有约束失败都报错;否则某些模式下(如ALLOW_INVALID_DATES)可能静默截断或转为默认值

外键约束如何影响UPDATE/DELETE性能与锁行为

InnoDB中外键操作会自动加锁并触发额外查询,这是容易被低估的性能开销来源。例如对父表执行UPDATE主键值,不仅会锁住该行,还会对子表中所有匹配的外键行加S(共享)锁——即使你只改一个字段,也可能导致子表全表扫描(若子表外键列无索引)。

典型陷阱:ALTER TABLE child_table ADD CONSTRAINT fk_parent_id FOREIGN KEY (parent_id) REFERENCES parent(id) 执行成功,但没给child_table.parent_id建索引,后续任何DELETE FROM parent WHERE id = ?都会触发子表全表扫描 + 行锁,拖慢整个事务。

  • 外键列必须有索引(单列或作为最左前缀),否则InnoDB拒绝创建外键(5.7+ 版本会报错 ERROR 1822 (HY000)
  • ON DELETE CASCADE看似方便,但大表级联删除可能长时间持有锁,且不记录单条日志,不利于binlog回放定位问题
  • 批量INSERT INTO ... SELECT含外键引用时,约束检查按行逐条进行,无法向普通INSERT那样用bulk insert buffer优化

延迟约束(DEFERRABLE)在MySQL中不存在

MySQL**不支持SQL标准中的延迟约束(DEFERRABLE INITIALLY DEFERRED)**。这意味着你无法把约束检查推迟到事务提交时才做——所有约束都在每条DML语句执行完毕后立即验证。这个限制直接影响复杂业务逻辑的设计方式。

citySHOP多用户商城系统
citySHOP多用户商城系统

citySHOP是一款集CMS、网店、商品、分类信息、论坛等为一体的城市多用户商城系统,已完美整合目前流行的Discuz! 6.0论坛,采用最新的5.0版PHP+MYSQL技术。面向对象的数据库连接机制,缓存及80%静态化处理,使它能最大程度减轻服务器负担,为您节约建设成本。多级店铺区分及联盟商户地图标注,实体店与虚拟完美结合。个性化的店铺系统,会员后台一体化管理。后台登陆初始网站密匙:LOVES

下载

比如想先插入子记录、再插入父记录(反向依赖),或在一个事务中交换两张表的主键值,MySQL天然不支持。常见变通做法是临时禁用约束检查,但必须极度谨慎:

  • SET FOREIGN_KEY_CHECKS = 0 只跳过外键检查,不影响主键/唯一/非空等其他约束
  • 该设置是会话级的,不会影响其他连接,但若在存储过程中使用,需确保异常路径也能恢复为1
  • 禁用后执行的非法数据(如孤立子记录)不会报错,但后续开启检查时也不会自动修复——它只影响“新写入”,不校验存量数据
SET FOREIGN_KEY_CHECKS = 0;
INSERT INTO child (id, parent_id) VALUES (1, 999); -- 此时不报错,即使parent_id=999不存在
INSERT INTO parent (id) VALUES (999);
SET FOREIGN_KEY_CHECKS = 1;

唯一索引NULL值处理与“伪重复”问题

MySQL中唯一索引(UNIQUE)对NULL值的处理是:**多个NULL被视为互不相等**。这常导致“看起来重复却通过校验”的情况,尤其在业务逻辑误将NULL当默认值时。

例如:CREATE TABLE user (email VARCHAR(255), UNIQUE(email)),允许插入多条email IS NULL的记录。但应用层若把未填邮箱统一设为NULL,就可能积累大量“空邮箱用户”,后续想补全时才发现违反唯一性。

  • 如果业务上NULL和空字符串''语义相同,应在应用或触发器中统一转为'',并让字段NOT NULL DEFAULT ''
  • 复合唯一索引中只要有一个列为NULL,整行即不参与重复判断(标准SQL行为,MySQL严格遵循)
  • 使用INSERT ... ON DUPLICATE KEY UPDATE时,若冲突键含NULL,不会触发UPDATE分支,因为NULL = NULLUNKNOWN,不构成“重复键匹配”

约束不是执行流程末端的“验收关卡”,而是嵌在解析、优化、执行各环节里的主动拦截器。真正难处理的从来不是报错本身,而是那些没报错却埋下数据歧义的场景——比如NULL在唯一索引里的自由穿梭,或者外键缺失索引导致的隐式全表扫描。

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

679

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

320

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

346

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1095

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

357

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

675

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

574

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

415

2024.04.29

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 1.8万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 793人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号