备注:
MySQL版本 5.7
Table of Contents
一. 问题描述
今天帮朋友写了一个存储过程,通过主键id分批更新一个大表,里面的update及delete语句大概有30多个,开启事务,一个批次更新完成后,再提交。
突然存储过程运行的时候出现如下报错:
2021-12-18T22:24:05.171910-05:00 22592 [ERROR] Event Scheduler: [root@localhost][**.****] Incorrect DECIMAL value: '0' for column '' at row -1 2021-12-18T22:24:05.171924-05:00 22592 [ERROR] Event Scheduler: [root@localhost][**.****] Truncated incorrect DECIMAL value: 'N' 2021-12-18T22:24:05.171930-05:00 22592 [ERROR] Event Scheduler: [root@localhost][**.****] Incorrect DECIMAL value: '0' for column '' at row -1 2021-12-18T22:24:05.171936-05:00 22592 [ERROR] Event Scheduler: [root@localhost][**.****] Truncated incorrect DECIMAL value: 'N' 2021-12-18T22:24:05.171940-05:00 22592 [ERROR] Event Scheduler: [root@localhost][**.****] Incorrect DECIMAL value: '0' for column '' at row -1 2021-12-18T22:24:05.171945-05:00 22592 [ERROR] Event Scheduler: [root@localhost][**.****] Truncated incorrect DECIMAL value: 'N'
复制
二. 解决方案
2.1 修改sql_mode
将配置文件sql_mode修改如下:
sql_mode="NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION"
复制
然后重启msyql服务,重试,问题依旧。
2.2 修改SQL语句
这个问题最坑之处在于,没有报具体是哪个sql语句的问题,给出的是一个模糊的报错。
首先我们查看表结果,发现没有decimal类型的字段,字段都是int或varchar类型,这就尴尬了。
后面全部复盘了一次修改的update语句,发现有个地方使用到rand()来生成随机数,虽然后floor函数处理成了整数。
UPDATE tab_name SET MM=FLOOR( 1 + RAND() * (12 - 1) WHERE MM is NULL and data_id between l_beg and l_end;
UPDATE tab_name SET MM=concat('0',mm) WHERE LENGTH(mm)=1 and data_id between l_beg and l_end;
UPDATE tab_name SET dd=FLOOR( 1 + RAND() * (28 - 1) WHERE dd is NULL and data_id between l_beg and l_end;
UPDATE tab_name SET dd=concat('0',dd) WHERE LENGTH(dd)=1 and data_id between l_beg and l_end;
复制
MM 和 DD 两列都是varchar类型, 这个业务的逻辑是,如果 MM和DD是空值的话,就刷成符合规范的随机数。
将rand() 生成的数值类型通过case函数转变为字符类型后,问题解决。
UPDATE tab_name SET MM=cast(FLOOR( 1 + RAND() * (12 - 1)) as char) WHERE MM is NULL and data_id between l_beg and l_end;
UPDATE tab_name SET MM=concat('0',mm) WHERE LENGTH(mm)=1 and data_id between l_beg and l_end;
UPDATE tab_name SET dd=cast(FLOOR( 1 + RAND() * (28 - 1)) as char) WHERE dd is NULL and data_id between l_beg and l_end;
UPDATE tab_name SET dd=concat('0',dd) WHERE LENGTH(dd)=1 and data_id between l_beg and l_end;
复制
「喜欢这篇文章,您的关注和赞赏是给作者最好的鼓励」
关注作者
【版权声明】本文为墨天轮用户原创内容,转载时必须标注文章的来源(墨天轮),文章链接,文章作者等基本信息,否则作者和墨天轮有权追究责任。如果您发现墨天轮中有涉嫌抄袭或者侵权的内容,欢迎发送邮件至:contact@modb.pro进行举报,并提供相关证据,一经查实,墨天轮将立刻删除相关内容。
文章被以下合辑收录
评论
相关阅读
【MySQL 30周年庆】MySQL 8.0 OCP考试限时免费!教你免费领考券
墨天轮小教习
2682次阅读
2025-04-25 18:53:11
MySQL 30 周年庆!MySQL 8.4 认证免费考!这次是认真的。。。
严少安
798次阅读
2025-04-25 15:30:58
【活动】分享你的压箱底干货文档,三篇解锁进阶奖励!
墨天轮编辑部
451次阅读
2025-04-17 17:02:24
MySQL 9.3 正式 GA,我却大失所望,新特性亮点与隐忧并存?
JiekeXu
411次阅读
2025-04-15 23:49:58
3月“墨力原创作者计划”获奖名单公布
墨天轮编辑部
368次阅读
2025-04-15 14:48:05
openHalo问世,全球首款基于PostgreSQL兼容MySQL协议的国产开源数据库
严少安
360次阅读
2025-04-07 12:14:29
记录MySQL数据库的一些奇怪的迁移需求!
陈举超
263次阅读
2025-04-15 15:27:53
MySQL 8.0 OCP 1Z0-908 考试解析指南(二)
JiekeXu
240次阅读
2025-04-30 17:37:37
MySQL 8.4 新特性深度解析:功能增强、废弃项与移除项全指南
JiekeXu
219次阅读
2025-04-18 20:21:32
SQL优化 - explain查看SQL执行计划(下)
金同学
217次阅读
2025-05-06 14:40:00