目录
一. 数值类型
1.1 整形类型
1.2 bit类型
1.3 小数类型
二. 字符串类型
2.1 char类型
2.2 varchar类型
三. 日期和时间类型
四. 枚举和集合类型
4.1 enum类型
4.2 set类型
五. 总结
数值类型,包括整形类型、bit类型和浮点数(小数)类型,表1给出了MySQL所在支持的所有数值类型,MySQL中的数值类型与C/C++中的int/double类型数据相似,每种数值类型都有其能表示的数值范围,并占用特定大小的空间,在使用数据库时应当根据具体的场景来选择数据类型。
分类 | 数值类型 | 大小(字节) | 说明 |
---|---|---|---|
整数类型 | tinyint | 1 | 无符号数范围是[0,2^8-1],有符号数范围是[-2^7, 2^7-1] |
samllint | 2 | 无符号数范围是[0,2^16-1],有符号数范围是[-2^15, 2^15-1] | |
mediumint | 3 | 无符号数范围是[0,2^24-1],有符号数范围是[-2^23, 2^23-1] | |
int | 4 | 无符号数范围是[0,2^32-1],有符号数范围是[-2^31, 2^31-1] | |
bigint | 8 | 无符号数范围是[0,2^64-1],有符号数范围是[-2^63, 2^63-1] | |
bit类型 | bit (M) | / | 类似与C语言中的位段类型,M表示有多少个bit位 |
小数类型 | float | 4 | 浮点数类型,能够表示最高约小数点后7位精度 |
double | 8 | 浮点数类型,与float类似,能够表示的数值范围高于float | |
decimal | / | 能够准确表述约小数点后约65位的精度 |
本文以tinyint类型为例,讲解MySQL中整形的使用方法及相关特性,下面的代码定义了一张名为t1的表,其中包含有符号tinyint类型成员num,尝试插入位于[-128,127]范围内的数据,可以成功插入。但是,如果插入超过tinyint所能表示的范围的数据,那么就插入失败了,MySQL会对超过表示类型范围的数据进行拦截,禁止插入。
结论:MySQL会拦截非法数据的插入,如超过特定类型能表示范围的数据。
mysql> create table t1 (
-> num tinyint
-> );
Query OK, 0 rows affected (0.16 sec)
mysql> insert into t1 values (122);
Query OK, 1 row affected (0.01 sec)
mysql> insert into t1 values (-128);
Query OK, 1 row affected (0.04 sec)
mysql> insert into t1 values (127); -- 插入位于tinyint表示范围内的数据插入成功
Query OK, 1 row affected (0.03 sec)
mysql> insert into t1 values (128); -- 超过tinyint类型表示范围的数据插入失败
ERROR 1264 (22003): Out of range value for column 'num' at row 1
mysql> insert into t1 values (-133);
ERROR 1264 (22003): Out of range value for column 'num' at row 1
mysql> select * from t1;
+------+
| num |
+------+
| 122 |
| -128 |
| 127 | -- 非法数据没有出现在表中
+------+
3 rows in set (0.01 sec)
下面的代码定义了表t2,将t1中的有符号tinyint成员num的类型转变为无符号tinyint unsigned类型,此时向t2中插入位于[128,255]范围内的数据全部插入成功,但如果试图插入负数或者超过256的数据,则依然会插入失败。
无符号数,相对于有符号,能够表示更大的非负数范围,但无法表示负数。
mysql> create table t2 (
-> num tinyint unsigned
-> );
Query OK, 0 rows affected (0.17 sec)
mysql> insert into t2 values (128);
Query OK, 1 row affected (0.03 sec)
mysql> insert into t2 values (256); -- 超过255的数据插入失败
ERROR 1264 (22003): Out of range value for column 'num' at row 1
mysql> insert into t2 values (255);
Query OK, 1 row affected (0.02 sec)
mysql> insert into t2 values (-130); -- 负数插入失败
ERROR 1264 (22003): Out of range value for column 'num' at row 1
bit类型,类似于C/C++中的位段,该成员的大小以二进制bit位来表示,如指定bit (1),那么表示一个单比特位数据成员,只能表示0和1。
语法:bit (M)
说明:M为比特位的个数,范围1~64,如果忽略则默认为1
在建表的时候,如果给出的M值超过64,会直接报错,MySQL会检查M不能超过bit类型的界限,如果企图给bit (M)类型数据赋予超过表达能力范围的数据,那么MySQL也会拦截操作。
mysql> create table t3 (
-> bnum bit (3)
-> );
Query OK, 0 rows affected (0.20 sec)
mysql> create table t4 (
-> bnum bit (64)
-> );
Query OK, 0 rows affected (0.17 sec)
mysql> create table t5 (
-> bnum bit (65) -- 超过bit类型二进制位的上限,被MySQL拦截
-> );
ERROR 1439 (42000): Display width out of range for column 'bnum' (max = 64)
mysql> insert into t3 values (8); -- 企图给bit(3)类型数据赋予超过表示能力范围的数据,被MySQL拦截
ERROR 1406 (22001): Data too long for column 'bnum' at row 1
mysql> insert into t3 values (7);
Query OK, 1 row affected (0.01 sec)
下面的代码创建了表t6,其中包含成员bnum,类型为bit (20),线向其中插入数据10和65,并通过select * from t6 显示表中的全部数据值,我们看到,10没有被显示出来,而65显示为'A',这说明bit (M) 类型数据在前端的默认显示的是其ASCII码对应的字符。
结论:bit (M) 在前端默认显示为其ASCII码对应的字符。
mysql> create table t6 (
-> bnum bit (20)
-> );
Query OK, 0 rows affected (0.22 sec)
mysql> insert into t6 values (10);
Query OK, 1 row affected (0.04 sec)
mysql> insert into t6 values (65);
Query OK, 1 row affected (0.03 sec)
mysql> select * from t6;
+------+
| bnum |
+------+
|
|
| A |
+------+
2 rows in set (0.00 sec)
首先介绍float类型,double类型与float类型的使用方式基本一致。
语法:float (M, D) unsigned
解释:M表示float类型数据最多能显示的有效数据长度(包括整数部分和小数部分),D表示精度为小数点后几位,必须要满足M ≥ N,unsigned表示无符号,可以省略。
对于float和double的无符号类型,与有符号的区别仅仅是不能表示有符号类型的负数部分,而正数部分并不能比对应有符号表示更广的范围,因此float和double一般都设置为有符号,但这也不绝对,有些场景下就是应当限制数值不为负数,这种情况下也应当声明unsigned。
下面的代码定义了float (4,2)类型的数据f1,其所能够表示的范围是-99.99~99.99,如果尝试插入超过范围的数据,那么就会报错。
mysql> create table t7 (
-> f1 float(4,2)
-> );
Query OK, 0 rows affected (0.14 sec)
mysql> insert into t7 values (-99.99); -- 不超过范围的数据插入成功
Query OK, 1 row affected (0.01 sec)
mysql> insert into t7 values (99.99);
Query OK, 1 row affected (0.02 sec)
mysql> insert into t7 values (12.37);
Query OK, 1 row affected (0.03 sec)
mysql> select * from t7;
+--------+
| f1 |
+--------+
| -99.99 |
| 99.99 |
| 12.37 |
+--------+
3 rows in set (0.00 sec)
mysql> insert into t7 values (123.23); -- 超过范围的数据插入失败
ERROR 1264 (22003): Out of range value for column 'f1' at row 1
mysql> insert into t7 values (-110.12);
ERROR 1264 (22003): Out of range value for column 'f1' at row 1
mysql> insert into t7 values (-100.00);
ERROR 1264 (22003): Out of range value for column 'f1' at row 1
mysql> select * from t7;
+--------+
| f1 |
+--------+
| -99.99 |
| 99.99 |
| 12.37 |
+--------+
3 rows in set (0.00 sec)
对于float (4,2)类型数据,如果试图插入的数据小数点后面超过了2位,那么所采取的策略是四舍五入,四舍五入也会受到范围的限制,如果舍入后在范围内,那么插入数据成功,如果在范围之外,那么插入失败。
如:尝试给float (4,2)插入99.994,插入成功,因为99.994舍入后的值是99.99未越界,如果尝试插入99.995,那么插入失败,因为舍入后的值为100,超过了范围。
mysql> create table t8 (
-> f1 float(4,2)
-> );
Query OK, 0 rows affected (0.20 sec)
mysql> insert into t8 values (12.123); -- 向下取值,实际插入12.12
Query OK, 1 row affected (0.04 sec)
mysql> insert into t8 values (91.567); -- 向上取值,实际插入91.57
Query OK, 1 row affected (0.02 sec)
mysql> insert into t8 values (-91.567); -- 向上取值,实际插入-91.57
Query OK, 1 row affected (0.02 sec)
mysql> insert into t8 values (99.994); -- 向下取值得99.99,不越界,插入成功
Query OK, 1 row affected (0.05 sec)
mysql> insert into t8 values (99.995); -- 向上取值得100.00,越界,插入失败
ERROR 1264 (22003): Out of range value for column 'f1' at row 1
mysql> select * from t8;
+--------+
| f1 |
+--------+
| 12.12 |
| 91.57 |
| -91.57 |
| 99.99 |
+--------+
4 rows in set (0.01 sec)
decimal类型数据的使用方法与float类型数据基本一致,decimal和float类型的区别在于:decimal相比于float能够表示更高的小数点后精度。
语法:decimal (M, D) unsigned
解释:M表示decimal类型数据最多能显示的有效数据长度(包括整数部分和小数部分),D表示精度为小数点后几位,必须要满足M ≥ N,unsigned表示无符号,可以省略。
下面的程序对float和decimal的精度进行了测试,创建一张表,其中包含两个数据f1和d1,f1的类型为float (10,8),d1的类型为decimal (10,8),向表中插入数据,f1和d1的值全部设为10.12345678,可以看到d1对应的decimal(10,8)类型正确存储了数据,而f1对应的float (10,8)出现了数据失真。
float最多能准确表示小数点后约7为数字,decimal能够准确表示约小数点后约65位。在对精度要求极高的业务场景中(如银行金融系统),应当采用decimal类型数据。
mysql> create table t9 (
-> f1 float(10,8),
-> d1 decimal(10,8)
-> );
Query OK, 0 rows affected (0.21 sec)
mysql> insert into t9 (f1,d1) values (10.12345678, 10.12345678);
Query OK, 1 row affected (0.03 sec)
mysql> select * from t9;
+-------------+-------------+
| f1 | d1 |
+-------------+-------------+
| 10.12345695 | 10.12345678 | -- decimal(10,8)准确表示了数据,而float(10,8)出现了失真
+-------------+-------------+
1 row in set (0.01 sec)
在MySQL中,字符串类型数据包括固定长度字符串类型char和变长字符串类型varchar。
类型 | 说明 |
---|---|
char (L) | 固定长度字符串,L表示最大存储字符数,L不超过255 |
varchar (L) | 可变长度字符串,L表示最大存储字符数,varchar类型最大不能超过65535字节 |
语法:char (L)
解释:能够存储最多L个字符的定长字符串类型数据。
下面的代码在表中定义了char (2)类型的成员msg,先后向表中插入 'ab' 和 ‘你好’,均能够插入成功,这里要注意区分MySQL层面和语言层面对字符的理解,在语言层面,一个字符占用1bytes的存储空间,utf8格式编码一个汉字要占用3个字节。在MySQL中,无论是一个汉字,还是一个C语言字符,都被解释为一个字符。
结论:在MySQL中,无论是一个汉字还是一个语言层面的字符,都被解释为一个字符。
mysql> create table t10 (
-> msg char(2)
-> );
Query OK, 0 rows affected (0.08 sec)
mysql> insert into t10 values ('10');
Query OK, 1 row affected (0.04 sec)
mysql> insert into t10 values ('ab');
Query OK, 1 row affected (0.03 sec)
mysql> insert into t10 values ('你好');
Query OK, 1 row affected (0.04 sec)
mysql> insert into t10 values ('abc');
ERROR 1406 (22001): Data too long for column 'msg' at row 1
mysql> insert into t10 values ('早上好');
ERROR 1406 (22001): Data too long for column 'msg' at row 1
语法:varchar (L)
解释:最多能够存储L个字符的变长字符串。
对于varchar类型的数据,其最大的长度为65535个字节,注意是65535字节而不是字符串,同时还需要1~3个字节来存储varchar类型数据的实际长度。
对应使用utf8编码的数据库(表),一个字符要占用3个字节,那么一个varchar类型数据最多能够存放 (65535 - 3) / 3 = 21844 个字符,对于使用gbk编码的数据库,一个字符占用2个字节,那么一个varchar类型数据最多能存放 (65535 - 3) / 2 = 32766 个字符。如果定义的varchar类型数据企图包含超过上限的字符数,那么MySQL会拦截操作。
varchar类型和char类型的对比:
mysql> create table t11 (
-> num varchar(21844)
-> )charset=utf8;
Query OK, 0 rows affected (0.16 sec)
mysql> create table t12 (
-> num varchar(21845) -- utf8编码下varchar超过21844个字符,插入失败
-> )charset=utf8;
ERROR 1118 (42000): Row size too large. The maximum row size for the used table type, not counting BLOBs, is 65535. This includes storage overhead, check the manual. You have to change some columns to TEXT or BLOBs
mysql>
mysql> create table t12 (
-> num varchar(32766)
-> )charset=gbk;
Query OK, 0 rows affected (0.08 sec)
mysql> create table t13 (
-> num varchar(32767) -- gbk编码下varchar超过21844个字符,插入失败
-> )charset=gbk;
ERROR 1118 (42000): Row size too large. The maximum row size for the used table type, not counting BLOBs, is 65535. This includes storage overhead, check the manual. You have to change some columns to TEXT or BLOBs
在MySQL中,日期和时间类型数据有如下三种:
如下代码所示,date和datetime类型数据需要用户手动插入,用于标识某个特定的时间,对于时间戳timestamp,每次在对表的某一行进行操作后,对应的时间戳timestamp类型数据都会自动被修改,时间戳是一种会动态变化的数据类型。
mysql> select * from t13;
+----+------------+---------------------+---------------------+
| id | d1 | d2 | d3 |
+----+------------+---------------------+---------------------+
| 1 | 1999-10-15 | 2011-12-08 12:08:23 | 2023-12-17 10:17:22 |
| 2 | 2003-07-22 | 2008-09-08 09:08:13 | 2023-12-17 10:17:55 |
+----+------------+---------------------+---------------------+
2 rows in set (0.00 sec)
mysql> update t13 set d1='2022-11-11' where id=1;
Query OK, 1 row affected (0.00 sec)
Rows matched: 1 Changed: 1 Warnings: 0
mysql> select * from t13;
+----+------------+---------------------+---------------------+
| id | d1 | d2 | d3 |
+----+------------+---------------------+---------------------+
| 1 | 2022-11-11 | 2011-12-08 12:08:23 | 2023-12-17 10:19:21 |
| 2 | 2003-07-22 | 2008-09-08 09:08:13 | 2023-12-17 10:17:55 |
+----+------------+---------------------+---------------------+
2 rows in set (0.00 sec)
语法:enum ('选项1', '选项2', '选项3', ... ...)
解释:enum与C/C++中的枚举类型类似,向枚举类型插入数据时,只能进行单选。
如下代码,表stu1定义了enum类型数据gender,包含两个选项 '男' 和 '女',对于enum类型数据,有以下两种方法可以插入数据:
当对枚举类型值进行修改和检索操作时,同样也可以使用直接给值和下标两种方式修改。
mysql> create table stu1 (
-> name varchar(20) not null,
-> gender enum('男', '女') -- 枚举类型数据,单选插入
-> );
Query OK, 0 rows affected (0.23 sec)
mysql> insert into stu1 values ('zhangsan','男'); -- 直接给定值插入枚举类型数据
Query OK, 1 row affected (0.03 sec)
mysql> insert into stu1 values ('lisi','女');
Query OK, 1 row affected (0.05 sec)
mysql> insert into stu1 values ('wangwu',1); -- 通过下标插入枚举类型数据
Query OK, 1 row affected (0.04 sec)
mysql> insert into stu1 values ('zhaoliu',2);
Query OK, 1 row affected (0.04 sec)
mysql> select * from stu1;
+----------+--------+
| name | gender |
+----------+--------+
| zhangsan | 男 |
| lisi | 女 |
| wangwu | 男 |
| zhaoliu | 女 |
+----------+--------+
4 rows in set (0.00 sec)
语法:set ('选项1', '选项2', '选项3', ... ...)
解释:set为集合类型,在插入的时候,可以选取其中的多个选项插入。
下面的代码定义了stu2类型数据,其中包含了 hobby set('编程', '绘画', '游泳', '篮球', '跑步'),对于set类型数据,通过直接给定数值,可以成功插入set类型数据。如果企图插入set中不存在的选项,那么MySQL会直接拦截操作。
set与enum的不同:
mysql> create table stu2 (
-> name varchar(20) not null,
-> hobby set('编程','绘画','游泳','篮球','跑步')
-> );
Query OK, 0 rows affected (0.12 sec)
mysql> insert into stu2 values ('zhang', ''); -- set类型数据可以不选取任何一个选项
Query OK, 1 row affected (0.01 sec)
mysql> insert into stu2 values ('zhao', '篮球');
Query OK, 1 row affected (0.03 sec)
mysql> insert into stu2 values ('qian', '跑步');
Query OK, 1 row affected (0.00 sec)
mysql> insert into stu2 values ('qian', '编程');
Query OK, 1 row affected (0.03 sec)
mysql> insert into stu2 values ('sun', '绘画,篮球');
Query OK, 1 row affected (0.03 sec)
mysql> insert into stu2 values ('li', '绘画,篮球,跑步');
Query OK, 1 row affected (0.01 sec)
mysql> select * from stu2;
+-------+----------------------+
| name | hobby |
+-------+----------------------+
| zhang | |
| zhao | 篮球 |
| qian | 跑步 |
| qian | 编程 |
| sun | 绘画,篮球 |
| li | 绘画,篮球,跑步 |
+-------+----------------------+
6 rows in set (0.00 sec)
mysql> insert into stu2 values ('li', '写作'); -- 不能插入set中不存在的选项
ERROR 1265 (01000): Data truncated for column 'hobby' at row 1
mysql> insert into stu2 values ('li', '编程,写作');
ERROR 1265 (01000): Data truncated for column 'hobby' at row 1
set类型数据,也可以通过数值的方式来插入,但是这时的数值不表示下标,而是以二进制位1/0表示set的每个选项是否被选择。以上面代码中创建的表stu2的hobby set('编程', '绘画', '游泳', '篮球', '跑步')类型数据为例,二进制位从低到高分别表示 '编程' ~ '跑步',如图4.1所示,5的二进制表示为00101,对应选择的就是 '编程' 和 '游泳'。
mysql> insert into stu2 values ('zhou',1);
Query OK, 1 row affected (0.03 sec)
mysql> insert into stu2 values ('wu',1);
Query OK, 1 row affected (0.04 sec)
mysql> insert into stu2 values ('zheng',5);
Query OK, 1 row affected (0.05 sec)
mysql> insert into stu2 values ('wang',7);
Query OK, 1 row affected (0.01 sec)
mysql> insert into stu2 values ('yang',8);
Query OK, 1 row affected (0.04 sec)
mysql> insert into stu2 values ('yang',16);
Query OK, 1 row affected (0.03 sec)
mysql> select * from stu2;
+-------+----------------------+
| name | hobby |
+-------+----------------------+
| zhang | |
| zhao | 篮球 |
| qian | 跑步 |
| qian | 编程 |
| sun | 绘画,篮球 |
| li | 绘画,篮球,跑步 |
| zhou | 编程 |
| wu | 编程 |
| zheng | 编程,游泳 |
| wang | 编程,绘画,游泳 |
| yang | 篮球 |
| yang | 跑步 |
+-------+----------------------+
12 rows in set (0.01 sec)
对于set类型的查找,如果通过where XXX=?的方式来进行查找,则查找的只能是完全匹配的数据,如下代码中,调用select * from stu2 where hobby='编程',找出来的是hobby仅为编程的行数据,如果hobby中包含'编程',但是又有其他的选项,那么则无法查找。
mysql> select * from stu2 where hobby='编程';
+------+--------+
| name | hobby |
+------+--------+
| qian | 编程 |
| zhou | 编程 |
| wu | 编程 |
+------+--------+
3 rows in set (0.00 sec)
为了克服上面的问题,应当使用MySQL库内置的子集查找函数find_in_set来进行查找。
函数原型:find_in_set(sub, setList)
说明:在setList中查找选项sub,如果sub在setList中存在,那么返回sub在setList中的下标,如果不存在,那么就返回0。
通过find_in_set查询的语法为:
mysql> select * from stu2;
+-------+----------------------+
| name | hobby |
+-------+----------------------+
| zhang | |
| zhao | 篮球 |
| qian | 跑步 |
| qian | 编程 |
| sun | 绘画,篮球 |
| li | 绘画,篮球,跑步 |
| zhou | 编程 |
| wu | 编程 |
| zheng | 编程,游泳 |
| wang | 编程,绘画,游泳 |
| yang | 篮球 |
| yang | 跑步 |
+-------+----------------------+
12 rows in set (0.00 sec)
mysql> select * from stu2 where find_in_set('编程', hobby);
+-------+----------------------+
| name | hobby |
+-------+----------------------+
| qian | 编程 |
| zhou | 编程 |
| wu | 编程 |
| zheng | 编程,游泳 |
| wang | 编程,绘画,游泳 |
+-------+----------------------+
5 rows in set (0.00 sec)
mysql> select * from stu2 where find_in_set('编程',hobby) and find_in_set('绘画', hobby);
+------+----------------------+
| name | hobby |
+------+----------------------+
| wang | 编程,绘画,游泳 |
+------+----------------------+
1 row in set (0.00 sec)