1 MySQL数据类型
MySQL支持多种数据类型,主要有以下3类:
(1)数值数据类型:包括整数类型TINYINT、SMALLINT、MEDIUMINT、INT、BIGINT、浮点小数数据类型FLOAT和DOUBLE、定点小数类型DECIMAL。
(2)日期/时间类型:包括YEAR、TIME、DATE、DATETIME和TIMESTAMP。
(3)字符串类型:CHAR、VARCHAR、BINARY、VARBINARY、BLOB、TEXT、ENUM和SET等。
1.1 整数类型
数值型数据类型主要用来存储数字。整数类型的属性字段可以添加AUTO_INCREMENT自增约束条件。
类型名称 | 说明 | 存储需求(字节) | 有符号 | 无符号 |
TINYINT | 很小的整数 | 1 | -128~127 | 0~255 |
SMALLINT | 小的整数 | 2 | -32768~32767 | 0~65535 |
MEDIUMINT | 中等大小的整数 | 3 | -8388608~8388607 | 0~16777215 |
INT(INTEGER) | 普通大小的整数 | 4 | -2147483648~2147483647 | 0~4294967295 |
BIGINT | 大整数 | 8 | -9223372036854775808~9223372036854775807 | 0~18446744073709551615 |
注意,有如下建表语句:
CREATE TABLE students
(
id INT(11);
name VARCHAR(25)
);
id字段的数据类型为INT(11),注意数字11表示的是该数据类型指定的是显示宽度,指定能够显示的数值中数字的个数。
显示宽度和数据类型的取值范围无关。显示宽度只是指明MySQL最大可能显示的数字个数,数值的位数小于指定的宽度时会由空格填充;如果插入了大于显示宽度的值,只要该值不超过类型整数的取值范围,数值依然可以插入,而且能够显示出来。
如果建表时不指定显示宽度,则系统会默认指定。注意的是负号占了一个数字位。
1.2 浮点数类型和定点数类型
浮点类型有两种:单精度浮点类型(FLOAT)和双精度浮点类型(DOUBLE)。
定点类型只有一种:DECIMAL。
浮点类型和定点类型都可以用(M,N)来表示,其中M称为精度,表示总共的位数;N称为标度,是表示小数的位数。
类型名称 | 存储需求(字节) | 有符号取值范围 | 无符号取值范围 |
FLOAT | 4 | -3.402823466E+38~-1.175494351E-38 | 0和1.175494351E-38~3.402823466E+38 |
DOUBLE | 8 | -1.7976931348623157E+308~-2.2250738585072014E-308 | 0和2.2250738585072014E-308~1.7976931348623157E+308 |
1.3 日期和时间类型
类型名称 | 日期格式 | 日期范围 | 存储需求(字节) |
YEAR | YYYY | 1901~2155 | 1 |
TIME | HH:MM:SS | -838:59:59~838:59:59 | 3 |
DATE | YYYY-MM-DD | 1000-01-01~9999-12-3 | 3 |
DATETIME | YYYY-MM-DD HH:MM:SS | 1000-01-01 00:00:00~9999-12-31 23:59:59 | 8 |
TIMESTAMP | YYYY-MM-DD HH:MM:SS | 1970-01-01 00:00:01 UTC ~ 2038-01-19 03:14:07 UTC | 4 |
1.4 字符串类型
字符串类型用来存储字符串数据,除了可以存储字符串数据之外,还可以存储其他数据,比如图片和声音的二进制数据。
类型名称 | 说明 | 存储需求 |
CHAR(M) | 固定长度非二进制字符串 | M字节,1<=M<=255 |
VARCHAR(M) | 变长非二进制字符串 | L+1字节,在此L<=M和1<=M<=255 |
TINYTEXT | 非常小的非二进制字符串 | L+1字节,在此L<2^8 |
TEXT | 小的非二进制字符串 | L+2字节,在此L<2^16 |
MEDIUMTEXT | 中等大小的非二进制字符串 | L+3字节,在此L<2^24 |
LONGTEXT | 大的非二进制字符串 | L+4字节,在此L<2^32 |
ENUM | 枚举类型,只能有一个枚举字符串值 | 1或2个字节,取决于枚举值的数目(最大值65535) |
SET | 一个设置,字符串对象可以有零个或多个SET成员 | 1,2,3,4或8个字节,取决于集合成员的数量(最多64个成员) |
1.4.1 CHAR和VARCHAR类型
CHAR(M)为固定长度字符串,在定义时指定字符串列长。当保存时在右侧填充空格以达到指定的长度。
VARCHAR(M)是长度可变的字符串,M表示最大列长度。M的范围是0~65535.
1.4.2 TEXT类型
TEXT列保存非二进制字符串,如文章内容、评论等。当保存或查询TEXT列的值时,不删除尾部空格
(1)TINYTEXT最大长度为255(28-1)字符的TEXT列。
(2)TEXT最大长度为65535(216-1)字符的TEXT列。
(3)MEDIUMTEXT最大长度为16777215(224-1)字符的TEXT列
(4)LONGTEXT最大长度为4294967295或4GB(232-1)字符的TEXT列。
1.4.3 ENUM类型
ENUM是一个字符串对象,其值为表创建时在列规定中枚举的一列值。语法格式;
字段名 ENUM('值1','值2',...,'值n')
1.4.4 SET类型
SET是一个字符串对象,可以有零个或多个值,SET列最多可以有64个成员,其值为表创建时规定的一列值。语法格式:
字段名 SET('值1','值2',...,'值n')
1.5 二进制类型
类型名称 | 说明 | 存储需求(字节) |
BIT(M) | 位字段类型 | 大约(M+7)/8 |
BINARY(M) | 固定长度二进制字符串 | M个字节 |
VARBINARY(M) | 可变长度二进制字符串 | M+1个字节 |
TINYBLOB(M) | 非常小的BLOB | L+1字节,在此L<2^8 |
BLOB(M) | 小BLOB | L+2字节,在此L<2^16 |
MEDIUMBLOB(M) | 中等大小的BLOB | L+3字节,在此L<2^24 |
LONGBLOB(M) | 非常大的BLOB | L+4字节,在此L<2^32 |
2 如何选择数据类型
2.1 整数和浮点数
如果不需要小数部分,则使用整数来保存数据;如果需要表示小数部分,则使用浮点数类型。对于浮点数据列,存入的数值会对该列定义的小数位进行四舍五入。
浮点数类型中,DOUBLE类型精度比FLOAT类型高,因此,如果要求存储精度较高时,应选择DOUBLE类型。
2.2 浮点数和定点数
浮点数FLOAT,DOUBLE相对于定点数DECIMAL的优势是:在长度一定的情况下,浮点数能表示更大的数据范围。但是由于浮点数容易产生误差,因此对精确度要求比较高时,建议使用DECIMAL来存储。
DECIMAL在MySQL中是以字符串存储的,用于定义货币等对精确度要求较高的数据。在数据迁移中,float(M,N)是非标准的SQL定义,数据库迁移可能会出现问题,最好不好这样使用。
另外两个浮点数进行减法和比较运算时也容易出问题,因此在进行计算的时候,一定要小心。如果进行数值比较,最好使用DECIMAL类型。
2.3 日期和时间类型
如果只需要记录年份,则使用YEAR类型;如果只记录时间,则使用TIME类型;
如果同时记录日期和时间,则可以使用TIMESTAMP或者DATETIME类型。TIMESTAMP列的取值范围小于DATETIME的取值范围。
默认情况下,当插入一条记录但并没有制定TIMESTAMP这个列值时,MySQL会把TIMESTAMP列设为当前的时间。
2.4 CHAR和VARCHAR
CHAR和VARCHAR的区别:
CHAR是固定长度字符,VARCHAR是可变长度字符;CHAR会自动删除插入数据的尾部空格,VARCHAR不会删除尾部空格。
CHAR是固定长度,所以它的处理速度比VARCHAR的速度要快,但是它的缺点就是浪费存储空间。
存储引擎对于选择CHAR和VARCHAR的影响:
对于MyISAM存储引擎:最好使用固定长度的数据列代替可变长度的数据列。这样可以使整个表静态化,从而使数据检索更快,用空间换时间。
对于InnoDB存储引擎:使用可变长度的数据列,因为InnoDB数据表的存储格式不分固定长度和可变长度,因此食用CHAR不一定比使用VARCHAR更好,但由于VARCHAR是按照实际的长度存储,比较节省空间,所以对磁盘I/O和数据存储总量比较好。
2.5 ENUM和SET
ENUM只能取单值,它的数据列表是一个枚举集合。它的合法取值列表最多允许有65535个成员。比如:性别字段适合定义为ENUM类型,每次只能从“男”或“女”中取一个值。
SET可取多值。它的合法取值列表最多允许有64个成员。空字符串也是一个合法的SET值。比如:要存储一个人兴趣爱好,最好使用SET类型。
ENUM和SET的值是以字符串形式出现的,但在内部,MySQL以数值的形式存储。
2.6 BLOB和TEXT
BLOB是二进制字符串,TEXT是非二进制字符串,两者均可存放大容量的信息。BLOB主要存储图片、音频信息等,而TEXT只能存储纯文本文件。