MySQL 小数类型介绍

文章目录

    • 前言
      • 1. 浮点类型
        • 1.1 数值精度说明
        • 1.2 整数超出范围
        • 1.3 小数超出范围
        • 1.4 精度误差说明
      • 2. 定点类型
        • 2.1 数值精度说明
        • 2.2 整数超出范围
        • 2.3 小数超出范围
    • 总结

前言

对于保证精度的数字,MySQL 也有对应的小数类型,下图是 MySQL 中小数类型概览。
MySQL 小数类型介绍_第1张图片

  • 浮点:小数点非固定的数,可表示数据范围较广,整数,小数都可表示。

  • 定点:小数点固定,可表示整数,小数。int(整数)本质是小数点位于末尾的 32 位定点数而已。

    -- 建表
    create table test_1(
        f1 float(5,2),
        f2 decimal(5,2)
    );
    
    -- 写入
    insert into test_1 value(100, 100);
    select * from test_1;
    
    f1 f2
    100 100.00
  • 单精度:使用 4 个字节存储,有效数字为 8 位,MySQL 中的 float 类型为单精度。

  • 双精度:使用 8 个字节存储,有效数字为 16 位,MySQL 中的 double 类型为双精度。

    -- 建表,精度和标度,都是用 MySQL 中的极限值
    create table test_13(
    f1 float(255, 30) UNSIGNED,
    f2 DOUBLE(255, 30) UNSIGNED
    );
    
    -- 写入圆周率
    insert into test_13 value(3.14159265358979323846, 3.14159265358979323846);
    
    select * from test_5;
    
    f1 f2
    3.1415927 3.141592653589793

1. 浮点类型

MySQL 中的浮点类型有 float 和 double,其中 float 为单精度 double 为双精度。

1.1 数值精度说明

MySQL 允许使用非标准语法(其他数据库未必支持,因此如果涉及到数据迁移,则最好不要这么用)FLOAT(M,D) 或 DOUBLE(M,D)。这里,M 称为精度,D 称为标度。(M,D) 中 M = 整数位 + 小数位,D = 小数位。 D <= M <= 255,0 <= D <= 30。

例如,定义为 FLOAT(5,2) 的一个列可以显示为 -999.99 - 999.99 如果超过这个范围会报错。

浮点类型,也可以加 UNSIGNED,但是不会改变数据范围,例如:FLOAT(3,2) UNSIGNED 仍然只能表示 0 - 9.99 的范围。

PS:整数类型 UNSIGNED 和 SIGNED 的范围会有变化,例如 int 类型,UNSIGNED 的最大值是 4294967295 有 SIGNED 的最大值是 2147483647。

所以在 MySQL 8.0 版本,已经计划取消指定位数,及 UNSIGNED 小数类型,创建表时会有 warning:

Specifying number of digits for floating point data types is deprecated and will be removed in a future release.
UNSIGNED for decimal and floating point data types is deprecated and support for it will be removed in a future release.

1.2 整数超出范围

如果存储时,整数部分超出了范围,MySQL就会报错,不允许存这样的值。

-- 创建测试表,范围是 -999.9 - 999.9
CREATE TABLE test1
(
    f1 FLOAT(4, 1),
    f2 double(4, 1)
);

-- 插入整数范围内的数字,可以正常插入
insert into test1 value (123.4, 123.4);

-- 插入范围外的数字,异常
insert into test1 value (1234.5, 1234.5);
--  [22003][1264] Out of range value for column 'f1' at row 1

1.3 小数超出范围

如果小数超出范围,会四舍五入,如果四舍五入计算后,导致整数超出范围,会报错。

-- 创建测试表,范围是 -999.9 - 999.9
CREATE TABLE test1
(
    f1 FLOAT(4, 1),
    f2 double(4, 1)
);

-- 插入整数范围内的数字,可以正常插入,小数超过范围
insert into test1 value (123.45, 123.45);

-- 小数超过范围,四舍五入后,整数也会超过范围,返回异常
insert into test1 value (999.95, 999.95);
--  [22003][1264] Out of range value for column 'f1' at row 1
f1 f2
123.4 123.4
123.5 123.5

1.4 精度误差说明

-- 创建一张表,DOUBLE 类型
CREATE TABLE test_double2
(
    f1 DOUBLE
);

-- 插入数据
INSERT INTO test_double2
VALUES (0.47),
       (0.44),
       (0.19);

-- 计算
select sum(f1) from test_double2;
sum(f1)
1.0999999999999999

在计算时,可能会丢失精度,所以与钱相关的字段,建议使用 DECIMAL 类型。一般为 DECIMAL(16, 4) 精确到 角、分、厘、毫。

2. 定点类型

MySQL 中定点类型一般指的是 DECIMAL,列的声明语法是DECIMAL(M,D)。NUMERIC 与 DECIMAL 同义,如果字段类型定义为 NUMERIC,则将自动转成 DECIMAL。

2.1 数值精度说明

对于声明语法 DECIMAL(M,D),自变量的值范围如下:

  • M 是最大位数(精度),范围是 1 到 65。可不指定,默认值是 10。
  • D 是小数点右边的位数(小数位)。范围是 0 到 30,并且不能大于 M,可不指定,默认值是 0。

例如字段 salary DECIMAL(5,2),能够存储具有五位数字和两位小数的任何值,因此可以存储在 salary 列中的值的范围是从 -999.99 到 999.99

2.2 整数超出范围

如果存储时,整数部分超出了范围,MySQL就会报错,不允许存这样的值。

-- 创建测试表,范围是 -999.9 - 999.9
create table t2(
    f1 decimal(3, 2)
);

-- 插入整数范围内的数字,可以正常插入
insert into test1 value (3.14, 3.14);

-- 插入范围外的数字,异常
insert into test1 value (1234.5, 1234.5);
--  Out of range value for column 'f1' at row 1

2.3 小数超出范围

如果小数超出范围,会四舍五入,如果四舍五入计算后,导致整数超出范围,会报错。

-- 创建测试表,范围是 -999.9 - 999.9
create table t2(
    f1 decimal(3, 2)
);

-- 小数超过范围,会四舍五入自动截断
insert into t2 value(9.991);

-- 插入范围外的数字,异常
insert into t2 value(9.998);
--  Out of range value for column 'f1' at row 1
f1
9.99

总结

在 MySQL 中虽然 float、double、decimal 类型都可以存储小数,但是浮点类型 float、double 无法保证计算精度,和钱相关的字段建议使用 DECIMAL(16, 4) 精确到 角、分、厘、毫。

你可能感兴趣的:(MySQL,mysql,数据库)