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.13.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_columnROUND(float_column,1)double_columnROUND(double_column,1)decimal_columnROUND(decimal_column,1)
3.243.23.243.23.243.2
3.253.23.253.23.253.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_columnROUND(float_column,1)ROUND(CAST(float_column AS decimal(3,2)),1)CAST(float_column AS decimal(3,1))
3.243.23.23.2
3.253.23.33.3
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值