MySQL客户端执行SQL出错时,错误信息从哪来
MySQL客户端(如 mysql 命令行、Python的
pymysql、PHP的
mysqli)执行SQL失败时,错误不是由SQL本身“吐出来”的,而是服务端返回的协议包里带的错误码和消息。客户端收到后才抛异常或设错误标志。
这意味着:不主动检查返回值或捕获异常,错误就静默吞掉了。
mysql命令行默认开启
--force以外的行为:遇到错误直接停止后续语句(除非用
source批量执行且加了
--force) 在存储过程中用
INSERT INTO ... SELECT出错,默认中断整个过程,除非定义了
DECLARE ... HANDLER应用层如 Node.js 的
mysql2,必须监听
error事件或用
try/catch包裹
await query()
如何在存储过程中捕获并处理错误
MySQL 5.5+ 支持基于 SQLSTATE 或 MySQL 错误码的异常处理机制,核心是
DECLARE ... HANDLER,但仅限于存储过程、函数、触发器内使用,不能用于普通交互式SQL。
常见写法:
DROP PROCEDURE IF EXISTS safe_insert;
DELIMITER $$
CREATE PROCEDURE safe_insert(IN p_id INT)
BEGIN
DECLARE EXIT HANDLER FOR SQLEXCEPTION
BEGIN
ROLLBACK;
SELECT '发生未知错误,已回滚' AS msg;
END;
<p>DECLARE EXIT HANDLER FOR SQLSTATE '23000' -- 如主键冲突、唯一索引冲突
BEGIN
ROLLBACK;
SELECT '主键/唯一约束冲突' AS msg;
END;</p><p>START TRANSACTION;
INSERT INTO users(id, name) VALUES(p_id, 'test');
COMMIT;
END$$
DELIMITER ;
SQLSTATE '23000'覆盖大部分完整性约束错误;具体可查 MySQL 官方错误码表
EXIT HANDLER触发后自动退出当前作用域(即过程体),
CONTINUE HANDLER则继续往下执行 注意:事务状态不会自动恢复,
ROLLBACK必须显式写,否则可能留下部分提交的数据
应用代码中怎么可靠判断SQL是否执行失败
靠
mysql_affected_rows()或类似接口判断“影响行数”是常见误区——它对
SELECT返回 -1,对成功执行但没改数据的
UPDATE返回 0,但这两者都不是错误。真正要捕获的是执行层面的失败。 PHP
mysqli:检查
$mysqli->errno是否非零,或调用
$mysqli->error;面向对象风格下用
$mysqli->connect_errno和
$mysqli->errorPython
pymysql:必须用
try/except pymysql.Error as e:,
e.args是
(errno, error_message)元组,例如
(1062, "Duplicate entry '1' for key 'PRIMARY'")Node.js
mysql2:监听
connection.on('error', ...) 处理连接级错误;单条查询用 await connection.execute(...)并
catch错误对象,其
code字段是类似
ER_DUP_ENTRY的符号名,
errno是数字(如 1062) 命令行
mysql -e "...":检查进程退出码,非零即失败(但无法区分错误类型)
为什么 SET sql_mode 不是错误处理,而是预防手段
sql_mode控制的是“什么算错误”,而不是“怎么处理错误”。比如开启
STRICT_TRANS_TABLES后,向
NOT NULL字段插
NULL会报错;关闭它则静默转成默认值(如空字符串或 0)。
这看起来像“容错”,实则是掩盖问题源头:
INSERT INTO t(a) VALUES(NULL)在非严格模式下可能成功,但业务逻辑已偏离预期 不同环境(开发/生产)
sql_mode不一致,会导致相同SQL行为不一致,调试困难 它不能替代应用层的错误捕获——即使开了严格模式,主键冲突、锁等待超时等仍需按异常处理
真正容易被忽略的是:错误码的语义依赖 MySQL 版本和配置。比如
1205(死锁)在 5.7 和 8.0 中都存在,但
3819(check constraint violation)只在 8.0.16+ 支持。写通用错误处理时,别只认数字,优先匹配
SQLSTATE或错误消息子串。
