MySQL 浮点数字段“四舍五入”问题

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()函数有如下两条规则

  1. For exact-value numbers, ROUND() uses the “round half up”
    rule.
    对于精确的数值, ROUND 函数使用四舍五入
  2. 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.25ROUND(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
发布了27 篇原创文章 · 获赞 2 · 访问量 3508

猜你喜欢

转载自blog.csdn.net/tysite/article/details/93890007