MySQL的float数据类型注意

来源:互联网 发布:杨氏弹性模量数据ed 编辑:程序博客网 时间:2024/06/07 06:00

摘要:

公司项目数据库中存储金额字段使用float类型,在金额达到10W时出现小数丢失,部分数据四舍五入的问题,将数据类型替换为DECIMAL后解决该问题。主要原因就是原来人员没有搞清楚float的使用陷阱和使用场景,以下内容就是自己查找部分同学的资料后实验所得。


MySQL官方文档

FLOAT[(M[, D])]数据类型是单精度浮点数, 默认大小为24位数字,精度大约7位数字(经测试为6位),当设置M大小大于24时,自动转换为DOUBLE类型;同时设置M和D时不进行自动转换。


测试实验1

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.


测试实验2

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).


测试实验3

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;当小数位多时进行四舍五入.


测试实验4

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类型.


测试实验5

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

0 0
原创粉丝点击