公司项目数据库中存储金额字段使用float类型,在金额达到10W时出现小数丢失,部分数据四舍五入的问题,将数据类型替换为DECIMAL后解决该问题。主要原因就是原来人员没有搞清楚float的使用陷阱和使用场景,以下内容就是自己查找部分同学的资料后实验所得。
FLOAT[(M[, D])]数据类型是单精度浮点数, 默认大小为24位数字,精度大约7位数字(经测试为6位),当设置M大小大于24时,自动转换为DOUBLE类型;同时设置M和D时不进行自动转换。
mysql > create table float_test(id int, float_test float);
mysql > desc float_test;
+------------+---------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+------------+---------+------+-----+---------+-------+
| id | int(11) | YES | | NULL | |
| float_test | float | YES | | NULL | |
+------------+---------+------+-----+---------+-------+
mysql > insert into float_test values(1, 123456),(2, 123.456),(3, 1234567),(4,1234.5678);
mysql > select * from float_test;
+------+------------+
| id | float_test |
+------+------------+
| 1 | 123456 |
| 2 | 123.456 |
| 3 | 123570 |
| 4 | 1234.57 |
+------+------------+
从上面测试代码可知float默认保存6位精度(包括小数位和整数位),超过6位会被四舍五入并补入0.
mysql > alter table float_test add float_4 float(4);
mysql > desc float_test;
+------------+---------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+------------+---------+------+-----+---------+-------+
| id | int(11) | YES | | NULL | |
| float_test | float | YES | | NULL | |
| float_4 | float | YES | | NULL | |
+------------+---------+------+-----+---------+-------+
由数据表结构来看,float(m)是没有效果的,和float效果一致(根据文档m<24时相当于24).
mysql > alter table float_test add float_4_2 float(4,2);
mysql > desc float_test;
+------------+------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+------------+------------+------+-----+---------+-------+
| id | int(11) | YES | | NULL | |
| float_test | float | YES | | NULL | |
| float_4 | float | YES | | NULL | |
| float_4_2 | float(4,2) | YES | | NULL | |
+------------+------------+------+-----+---------+-------+
mysql > insert into float_test(id,float_4_2) values(5, 12),(6, 12.4), (7, 12.34);(填写其他整数部分位数大于2的均会报错,但使用ORM框架不会报错)
mysql > select * from float_test where id > 4;
+------+------------+---------+-----------+
| id | float_test | float_4 | float_4_2 |
+------+------------+---------+-----------+
| 5 | NULL | NULL | 12.00 |
| 6 | NULL | NULL | 12.40 |
| 7 | NULL | NULL | 12.34 |
+------+------------+---------+-----------+
mysql > update float_test set float_4_2=12.3456 where id = 5;
mysql> select * from float_test where id > 4
+------+------------+---------+-----------+
| id | float_test | float_4 | float_4_2 |
+------+------------+---------+-----------+
| 5 | NULL | NULL | 12.35 |
| 6 | NULL | NULL | 12.40 |
| 7 | NULL | NULL | 12.34 |
+------+------------+---------+-----------+
由上可知,设置m和d是有效果的,小数不够位补0;当小数位多时进行四舍五入.
mysql> alter table float_test add float_25 float(25);
mysql> desc float_test;
+------------+------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+------------+------------+------+-----+---------+-------+
| id | int(11) | YES | | NULL | |
| float_test | float | YES | | NULL | |
| float_4 | float | YES | | NULL | |
| float_4_2 | float(4,2) | YES | | NULL | |
| float_25 | double | YES | | NULL | |
+------------+------------+------+-----+---------+-------+
当m大于24时,数据类型转换为double类型.
mysql> alter table float_test add float25_2 float(25,2);
mysql> desc float_test;
+------------+-------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+------------+-------------+------+-----+---------+-------+
| id | int(11) | YES | | NULL | |
| float_test | float | YES | | NULL | |
| float_4 | float | YES | | NULL | |
| float_4_2 | float(4,2) | YES | | NULL | |
| float_25 | double | YES | | NULL | |
| float25_2 | float(25,2) | YES | | NULL | |
+------------+-------------+------+-----+---------+-------+
mysql> insert into float_test(id, float25_2) values(8, 12345678901234567890123.12);
mysql> select * from float_test where id > 7;
+------+------------+---------+-----------+----------+----------------------------+
| id | float_test | float_4 | float_4_2 | float_25 | float25_2 |
+------+------------+---------+-----------+----------+----------------------------+
| 8 | NULL | NULL | NULL | NULL | 12345679377913908000000.00 |
+------+------------+---------+-----------+----------+----------------------------+
有由上可知,当m和d同时设置时,尽管m大于24,但是不会进行转换;同时数值还会变得不准确,小数位被0填充。
注:
float的d最大限制不能超过30