MySQL 浮点数字段 “四舍五入”查询时,数据计算错误问题
1、MySQL “四舍五入” 运算函数
通常,我们在进行MySQL数据“四舍五入”查询时,会用到 ROUND(x,y)
函数 ,在直接使用时,其数据结果显示正常,如下所示:
SELECT ROUND(3.14,1),ROUND(3.15,1) FROM dual;
执行结果如下:
ROUND(3.14,1) | ROUND(3.15,1) |
---|---|
3.1 | 3.2 |
2、MySQL 浮点数字段小数“四舍五入”问题演示
首先,创建一个含有 float和double类型的两个浮点数字段和一个decimal类型的小数字段
CREATE TABLE `demo` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`float_column` FLOAT,
`double_column` DOUBLE,
`decimal_column` DECIMAL(5,2),
PRIMARY KEY (`id`)
) ENGINE=InnoDB CHARSET=utf8 COLLATE=utf8_bin;
然后,写入特定临界点数据
insert into demo values
(null,3.24,3.24,3.24),
(null,3.25,3.25,3.25);
最后,执行查询脚本
SELECT float_column,ROUND(float_column,1), double_column,ROUND(double_column,1),decimal_column,ROUND(decimal_column,1) FROM demo;
执行结果如下表所示:
float_column | ROUND(float_column,1) | double_column | ROUND(double_column,1) | decimal_column | ROUND(decimal_column,1) |
---|---|---|---|---|---|
3.24 | 3.2 | 3.24 | 3.2 | 3.24 | 3.2 |
3.25 | 3.2 | 3.25 | 3.2 | 3.25 | 3.3 |
从查询结果中可以看出,float_column 和 double_column 使用 ROUND()
函数“四舍五入”后的结果 是 错误
的
那么这个问题的原因出在哪里呢?
我们查看MySQL的官方文档,发现关于 ROUND()
函数有如下两条规则
- For exact-value numbers, ROUND() uses the “round half up”
rule.
对于精确的数值, ROUND 函数使用四舍五入
- For approximate-value numbers, the
result depends on the C library. On many systems, this means that
ROUND() uses the “round to nearest even” rule: A value with any
fractional part is rounded to the nearest even integer.
对于近似值,则依赖于底层的C函数库,在很多系统中 ROUND 函数会使用“取最近的偶数”的规则
由此可见:在float和double两个浮点数字段中存储的 3.25
的ROUND(X,1)
函数运算时,符合第二条规则,故返回了 最近偶数: 3.2
。
3、解决方案
方案一:可以在设计表结构的时候,对此类需要四舍五入的数值字段,设置为 DECIMAL
类型。
方案二:通过CAST()
函数,将浮点型字段,转换为 DECIMAL
格式,再进行ROUND
函数处理 。当需要处理的仅为单个表字段时,直接使用CAST()
函数即可完成“浮点数”转换为“小数” 及 “四舍五入”计算
示例如下:
SELECT float_column,ROUND(float_column,1),ROUND(CAST(float_column AS decimal(3,2)),1), CAST(float_column AS decimal(3,1)) FROM demo;
查询结果:
float_column | ROUND(float_column,1) | ROUND(CAST(float_column AS decimal(3,2)),1) | CAST(float_column AS decimal(3,1)) |
---|---|---|---|
3.24 | 3.2 | 3.2 | 3.2 |
3.25 | 3.2 | 3.3 | 3.3 |