MySQL之ROUND函数四舍五入的陷阱

在MySQL中,ROUND函数用于对查询结果进行四舍五入,不过最近使用ROUND函数四舍五入时意外发现并无预期的那样,本文将这一问题记录下来,以避免你们跟我同样犯一样的错误。mysql

问题描述

假如咱们有以下一个数据表test,建表语句以下git

CREATE TABLE test (
  id int(11) NOT NULL AUTO_INCREMENT,
  field1 bigint(10) DEFAULT NULL,
  field2 decimal(10,0) DEFAULT NULL,
  field3 int(10) DEFAULT NULL,
  field4 float(15,4) DEFAULT NULL,
  field5 float(15,4) DEFAULT NULL,
  field6 float(15,4) DEFAULT NULL,
  PRIMARY KEY (id)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

咱们建立了一个名为test的表,出了id字段以外还包含了多个字段,拥有这不一样的数据类型。咱们向这个表中插入一条数据github

INSERT INTO test (field1, field2, field3, field4, field5, field6) VALUE (100, 100, 100, 1.005, 3.5, 2.5);

插入以后表中的数据是这样的sql

mysql> select * from test;
+----+--------+--------+--------+--------+--------+--------+
| id | field1 | field2 | field3 | field4 | field5 | field6 |
+----+--------+--------+--------+--------+--------+--------+
|  1 |    100 |    100 |    100 | 1.0050 | 3.5000 | 2.5000 |
+----+--------+--------+--------+--------+--------+--------+
1 row in set (0.00 sec)

若是如今咱们执行下面这个SQL,你以为结果会是什么样的呢?数据库

SELECT
  round(field1 * field4),
  round(field2 * field4),
  round(field3 * field4),
  round(field1 * 1.005),
  round(field2 * 1.005),
  round(field3 * 1.005),
  round(field5),
  round(field6)
FROM test;

最初一直觉得这样的结果确定是都是101,由于上面这六个取值结果都是对100 * 1.005进行四舍五入,结果确定都是101才对,然后面两个确定是43才对,可是最终的结果倒是与设想的截然不同app

*************************** 1. row ***************************
round(field1 * field4): 100
round(field2 * field4): 100
round(field3 * field4): 100
 round(field1 * 1.005): 101
 round(field2 * 1.005): 101
 round(field3 * 1.005): 101
         round(field5): 4
         round(field6): 2
1 row in set (0.00 sec)

为何会这样?

一样是100*1.005,为何从数据库中的字段相乘获得的结果和直接字段与小数相乘获得的不同呢?函数

对这个问题百思不得其解,各类百度谷歌无果。。。没办法,还得靠本身,这个时候最有用的就是官网文档了,因而查询了mysql官方文档中关于ROUND函数的部分,其中包含下面两条规则this

  • For exact-value numbers, ROUND() uses the “round half up” rule(对于精确的数值,ROUND函数使用四舍五入)rest

  • 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函数会使用“取最近的偶数”的规则)code

经过这两条规则,咱们能够看出,因为咱们在使用两个字段相乘的时候,最终的结果是按照float类型处理的,而在计算机中float类型不是精确的数,所以处理结果会按照第二条来,而直接整数字段与1.005这样的小数运算的结果是由于两个参与运算的值都是精确数,所以按照第一条规则计算。从field5field6执行ROUND函数的结果能够明确的看确实是转换为了最近的偶数。

总结

从这个例子中能够看到,在MySQL中使用ROUND仍是要很是须要注意的,特别是当参与计算的字段中包含浮点数的时候,这个时候计算结果是不许确的。

本文将会持续修正和更新,最新内容请参考个人 GITHUB 上的 程序猿成长计划 项目,欢迎 Star,另外,求follow?。

相关文章
相关标签/搜索