目录
一、数据类型的分类
二、数值类型
1、tinyint相关类型
1.1创建有符号数值
1.2创建无符号数值(创建字段时在属性后加unsigned)
2、bit类型
3、浮点数类型
3.1float/double(略)
3.2decimal
4、字符串类型
4.1char类型(固定长度字符串)
4.2varchar(可变长度字符串)
5、日期类型和时间类型
6、枚举和集合
6.1enum的枚举值和set的位图结构
6.2使用find_in_set进行集合查询
类型 |
字节 |
最小值 |
最大值 |
TINYINT(无符号) |
1 |
0 |
255 |
TINYINT(有符号) |
-128 |
127 |
|
SMALLINT(无符号) |
2 |
0 |
65535 |
SMALLINT(有符号) |
-32768 |
32767 |
|
MEDIUMINT(无符号) |
3 |
0 |
16777215 |
MEDIUMINT(有符号) |
-8388608 |
8388607 |
|
INT(无符号) |
4 |
0 |
4294967295 |
INT(有符号) |
-2147483648 |
2147483647 |
|
BIGINT(无符号) |
8 |
0 |
18446744073709551615 |
BIGINT(有符号) |
-9223372036854775808 |
9223372036854775807 |
#创建表t1,默认是有符号的。取值范围[-128,127]
mysql> create table if not exists t1( num tinyint );
Query OK, 0 rows affected (0.25 sec)
#在表中插入-128,成功
mysql> insert into t1 values(-128);
Query OK, 1 row affected (0.03 sec)
#在表中插入超出范围的数据将失败
mysql> insert into t1 values (128);
ERROR 1264 (22003): Out of range value for column 'num' at row 1
#创建表t2,unsigned无符号。取值范围[0,255]
mysql> create table if not exists t2(
-> num tinyint unsigned
-> );
Query OK, 0 rows affected (0.24 sec)
MySQL中插入的数据不能越界。所以MySQL的数据类型本身也是一种约束。在选用数据类型的时候,按需选取。
bit[(M)] :位字段类型。M表示有几位,范围从1到64位。如果M被忽略,默认为1。
#创建带位类型的表t3
mysql> create table if not exists t3(
-> id int,
-> online bit(1)
-> );
Query OK, 0 rows affected (0.20 sec)
#插入数据
mysql> insert into t3 (id,online) values (123,1);
Query OK, 1 row affected (0.03 sec)
mysql> insert into t3 (id,online) values (124,0);
Query OK, 1 row affected (0.02 sec)
mysql> insert into t3 (id) values (125);
Query OK, 1 row affected (0.05 sec)
#打印t3
mysql> select * from t3;
+------+--------+
| id | online |
+------+--------+
| 123 | |
| 124 | |
| 125 | NULL |
+------+--------+
3 rows in set (0.00 sec)
打印t3发现位类型并没有被显示,这是因为bit字段在显示时,是按照ASCII码对应的值显示。如果真想看到,可以使用十六进制打印。
#十六进制打印位类型
mysql> select id,hex(online) from t3;
+------+-------------+
| id | hex(online) |
+------+-------------+
| 123 | 1 |
| 124 | 0 |
| 125 | NULL |
+------+-------------+
3 rows in set (0.00 sec)
浮点数会存在精度的丢失。
float[(m, d)] [unsigned] : M指定显示的总长度(整数+小数),d指定小数位数。
float(5,2) 表示的范围是 -999.99 ~ 999.99
float(5,2) unsigned 表示的范围 0 ~ 999.99
当用户小数个数给多了,将会四舍五入进行存储。float占用空间4个字节。
#创建带浮点数类型的表t3
mysql> create table if not exists t4(
-> id int,
-> salary float(4,2)
-> );
Query OK, 0 rows affected (0.24 sec)
观察哪些数据可以被插入:小数:float(4,2)表示的范围是-99.99 ~ 99.99,MySQL在保存值时会进行四舍五入。
#共4位,小数2位满足要求,可以插入
mysql> insert into t4 (id,salary) values (1,99.98);
Query OK, 1 row affected (0.03 sec)
#要求4位,小数要求2位。最后一位小数四舍五入后最终存储数据位99.98。,满足要求
mysql> insert into t4 (id,salary) values (1,99.980);
Query OK, 1 row affected (0.00 sec)
#四舍五入后为100.00。一共五位数字,不满足要求
mysql> insert into t4 (id,salary) values (1,99.996);
ERROR 1264 (22003): Out of range value for column 'salary' at row 1
#一共五位数字,不满足要求
mysql> insert into t4 (id,salary) values (1,100.92);
ERROR 1264 (22003): Out of range value for column 'salary' at row 1
无符号的浮点数:
#创建带无符号浮点数类型的表t3
mysql> create table if not exists t5(
-> id bigint,
-> salary float(4,2) unsigned
-> );
Query OK, 0 rows affected (0.33 sec)
#无符号浮点数插入负数是非法的
mysql> insert into t5 (id,salary) values (1,-0.001);
ERROR 1264 (22003): Out of range value for column 'salary' at row 1
mysql> insert into t5 (id,salary) values (1,-0.0001);
ERROR 1264 (22003): Out of range value for column 'salary' at row 1
decimal(m, d) [unsigned] : M指定显示的总长度(整数+小数),d指定小数位数。
decimal(5,2) 表示的范围是 -999.99 ~ 999.99
decimal(5,2) unsigned 表示的范围 0 ~ 999.99
decimal和float很像,表示的数据范围一样,但是精度不一样:
#t6是一张double和decimal对比的表
mysql> desc t6;
+-------+---------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+-------+---------------+------+-----+---------+-------+
| f1 | float(10,8) | YES | | NULL | |
| f2 | decimal(10,8) | YES | | NULL | |
+-------+---------------+------+-----+---------+-------+
2 rows in set (0.02 sec)
#在表中插入相同的数据
mysql> insert into t6 (f1,f2) values (23.12345612, 23.12345612);
Query OK, 1 row affected (0.04 sec)
#float类型存在精度丢失,decimal存什么就是什么(精度高)
mysql> select f1,f2 from t6;
+-------------+-------------+
| f1 | f2 |
+-------------+-------------+
| 23.12345695 | 23.12345612 |
+-------------+-------------+
2 rows in set (0.00 sec)
float表示的精度大约是7位。
decimal整数最大位数m为65。支持小数最大位数d是30。DECIMAL 类型占用的存储空间为 M+2 字节,其中 M 表示数字的精度。如果d被省略,默认为0.如果m被省略,默认是10。
建议:如果希望小数的精度高,推荐使用decimal或double而不是float。
char(L): 固定长度字符串,L是可以存储的长度,单位为字符(注意是字符,不是字节),最大长度值可以为255
#创建char类型的表t7
mysql> create table if not exists t7(
-> id int,
-> name char(2)
-> );
Query OK, 0 rows affected (0.26 sec)
#插入字符
mysql> insert into t7 (id,name) values (1,'a');
Query OK, 1 row affected (0.03 sec)
mysql> insert into t7 (id,name) values (1,'ab');
Query OK, 1 row affected (0.03 sec)
mysql> insert into t7 (id,name) values (1,'abc');
ERROR 1406 (22001): Data too long for column 'name' at row 1
mysql> insert into t7 (id,name) values (1,'喝');
Query OK, 1 row affected (0.04 sec)
mysql> insert into t7 (id,name) values (1,'喝饮');
Query OK, 1 row affected (0.04 sec)
mysql> insert into t7 (id,name) values (1,'喝饮料');
ERROR 1406 (22001): Data too long for column 'name' at row 1
mysql> select * from t7;
+------+--------+
| id | name |
+------+--------+
| 1 | a |
| 1 | ab |
| 1 | 喝 |
| 1 | 喝饮 |
+------+--------+
4 rows in set (0.00 sec)
在UTF8编码格式中,虽然汉字占3个字节,但是在MySQL中,char是不看字节数的,只看字符的个数。所以这里无论是插入字符还是汉字,都是限制两个字符以内。
varchar(L): 可变长度字符串,L表示字符长度,最大长度65535个字节(注意这里是字节,而char那边是字符!例如utf8中,单字符是3字节,那么L的最大长度为21844字符!计算方式见下方:)
关于varchar(L),L到底是多大,这个len值,和表的编码密切相关:
varchar长度可以指定为0到65535之间的值,但是有1 - 3 个字节用于记录数据大小,所以说有效字节数是65532。(如果表中仅有一个varchar字段,有效字节数是65532,如果还有其他的字段,那么varchar的实际有效字节将会比65532低一点)
当我们的表的编码是utf8时,varchar(n)的参数n最大值是65532/3=21844[因为utf中,一个字符占用3个字节],如果编码是gbk,varchar(n)的参数n最大是65532/2=32766(因为gbk中,一个字符占用2字节)。
#创建失败
mysql> create table if not exists t9( name varchar(65535) );
ERROR 1074 (42000): Column length too big for column 'name' (max = 21845); use BLOB or TEXT instead
#创建失败
mysql> create table if not exists t9( name varchar(21845) );
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> create table if not exists t9( name varchar(21844) );
Query OK, 0 rows affected (0.31 sec)
char和varchar的区别:
- char是固定长度字符串,类似C/C++中的数组的概念,例如char(6)表示开辟6个字符的大小的空间,没用完就是浪费。
- varchar是可变长度的字符串,例如varchar(6)表示最大长度为6个字符,如果用户仅使用了一个字符,那么varchar只会分配出一个字符的空间+额外1个字节(记录数据大小)用于存储。
char和varchar类型的选择:
- 如果数据确定长度都一样,就使用定长(char),比如:身份证,手机号,md5
- 如果数据长度有变化,就使用变长(varchar), 比如:名字,地址,但是你要保证最长的能存的进去。
- 定长的磁盘空间有时比较浪费,但是效率高。
- 变长的磁盘空间有时比较节省,但是效率低。
- 定长的意义是,直接开辟好对应的空间
- 变长的意义是,在不超过自定义范围的情况下,用多少,开辟多少。
常用的日期有如下三个:
- date :日期 'yyyy-mm-dd' ,占用三字节
- datetime 时间日期格式 'yyyy-mm-dd HH:ii:ss' 表示范围从 1000 到 9999 ,占用八字节
- timestamp :时间戳,从1970年开始的 yyyy-mm-dd HH:ii:ss 格式和 datetime 完全一致,占用四字节
#创建带有日期类型和时间类型的表t10
mysql> create table if not exists t10(
-> t1 date,
-> t2 datetime,
-> t3 timestamp
-> );
Query OK, 0 rows affected (0.26 sec)
#打印t10的属性
mysql> desc t10;
+-------+-----------+------+-----+-------------------+-----------------------------+
| Field | Type | Null | Key | Default | Extra |
+-------+-----------+------+-----+-------------------+-----------------------------+
| t1 | date | YES | | NULL | |
| t2 | datetime | YES | | NULL | |
| t3 | timestamp | NO | | CURRENT_TIMESTAMP | on update CURRENT_TIMESTAMP |
+-------+-----------+------+-----+-------------------+-----------------------------+
3 rows in set (0.01 sec)
#插入数据后,时间戳自动更新
mysql> insert into t10 (t1,t2) values ('2000-10-01','2011-01-05 08:00:58');
Query OK, 1 row affected (0.03 sec)
mysql> select* from t10;
+------------+---------------------+---------------------+
| t1 | t2 | t3 |
+------------+---------------------+---------------------+
| 2000-10-01 | 2011-01-05 08:00:58 | 2023-06-10 16:29:07 |
+------------+---------------------+---------------------+
1 row in set (0.01 sec)
#创建带有enum和set类型的表votes
mysql> create table if not exists votes(
-> usename varchar(30),
-> gender enum('男','女'),
-> hobby set('写代码','吃饭','睡觉')
-> );
Query OK, 0 rows affected (0.28 sec)
#在表中插入数据
mysql> insert into votes values ('张三','男','吃饭, 睡觉');
Query OK, 1 row affected (0.01 sec)
#其中enum也可以直接使用枚举值进行写入
mysql> insert into votes values ('李四','1','吃饭');
Query OK, 1 row affected (0.03 sec)
mysql> insert into votes values ('王五',2,'吃饭');
Query OK, 1 row affected (0.04 sec)
#set也可输入数字插入,数字代表位图
mysql> insert into votes values ('小黑',2,3);
Query OK, 1 row affected (0.06 sec)
mysql> select* from votes;
+---------+--------+------------------+
| usename | gender | hobby |
+---------+--------+------------------+
| 张三 | 男 | 吃饭,睡觉 |
| 李四 | 男 | 吃饭 |
| 王五 | 女 | 吃饭 |
| 小明 | NULL | NULL |
| 小黑 | 女 | 写代码,吃饭 |
+---------+--------+------------------+
enum:在所有枚举中选择一种进行插入。也可以直接使用枚举值,枚举值从1开始。
set:在集合中选择一种或多种存在的选项进行数据的插入。也可使用位图的形式进行数据插入,数字代表的意思如上图所示。
#查看数据库中所有的数据
mysql> select * from votes;
+---------+--------+------------------+
| usename | gender | hobby |
+---------+--------+------------------+
| 张三 | 男 | 吃饭,睡觉 |
| 李四 | 男 | 吃饭 |
| 王五 | 女 | 吃饭 |
| 小明 | NULL | NULL |
| 小红 | 男 | 吃饭 |
| 小黑 | 女 | 写代码,吃饭 |
+---------+--------+------------------+
6 rows in set (0.02 sec)
#筛选爱好为"吃饭"的数据,发现是精确筛选
mysql> select * from votes where hobby='吃饭';
+---------+--------+--------+
| usename | gender | hobby |
+---------+--------+--------+
| 李四 | 男 | 吃饭 |
| 王五 | 女 | 吃饭 |
| 小红 | 男 | 吃饭 |
+---------+--------+--------+
3 rows in set (0.00 sec)
在集合中筛选爱好为“吃饭”的数据,发现是精确匹配。在集合中匹配查找使用find_in_set函数。
mysql> select find_in_set('a','abc');
+------------------------+
| find_in_set('a','abc') |
+------------------------+
| 0 |
+------------------------+
1 row in set (0.00 sec)
mysql> select find_in_set('a','a,b,c');
+--------------------------+
| find_in_set('a','a,b,c') |
+--------------------------+
| 1 |
+--------------------------+
1 row in set (0.00 sec)
mysql> select find_in_set('a,b','a,b,c');
+----------------------------+
| find_in_set('a,b','a,b,c') |
+----------------------------+
| 0 |
+----------------------------+
1 row in set (0.00 sec)
当一个要查找的字段位于集合中时,find_in_set函数会返回其下标(非0),反之返回0。所以我们就可以在集合中筛选出爱好带有“吃饭”的人。
#在集合中筛选出爱好带有“吃饭”的人。
mysql> select* from votes where find_in_set('吃饭',hobby);
+---------+--------+------------------+
| usename | gender | hobby |
+---------+--------+------------------+
| 张三 | 男 | 吃饭,睡觉 |
| 李四 | 男 | 吃饭 |
| 王五 | 女 | 吃饭 |
| 小红 | 男 | 吃饭 |
| 小黑 | 女 | 写代码,吃饭 |
+---------+--------+------------------+
5 rows in set (0.00 sec)
也可以在集合中筛选出爱好既有“吃饭”,又有“写代码”人。使用and逻辑与进行筛选:
mysql> select* from votes where find_in_set('吃饭',hobby) and find_in_set('写代码',hobby);
+---------+--------+------------------+
| usename | gender | hobby |
+---------+--------+------------------+
| 小黑 | 女 | 写代码,吃饭 |
+---------+--------+------------------+
1 row in set (0.00 sec)
也可以使用位图结构进行多种爱好的搜索:
mysql> select* from votes where hobby=3;
+---------+--------+------------------+
| usename | gender | hobby |
+---------+--------+------------------+
| 小黑 | 女 | 写代码,吃饭 |
+---------+--------+------------------+
1 row in set (0.02 sec)