当前位置:   article > 正文

MySQL数据类型

MySQL数据类型

目录

1.数据类型分类

2.数值类型

2.1 tinyint

2.2 bit

2.3浮点类型

2.3.1 float

2.3.2 decimal

3.字符串类型

3.1 char

3.2 varchar

4.日期类型

5.枚举和集合类型


1.数据类型分类

2.数值类型

2.1 tinyint

  • 在MySQL中,整型可以指定是有符号的和无符号的,默认是有符号的。
  • 可以通过UNSIGNED来说明某个字段是无符号的
  1. mysql> create table t1( num tinyint );
  2. Query OK, 0 rows affected (0.02 sec)
  3. mysql> insert into t1 values(100);
  4. Query OK, 1 row affected (0.00 sec)
  5. mysql> insert into t1 values(127);
  6. Query OK, 1 row affected (0.00 sec)
  7. mysql> insert into t1 values(-128);
  8. Query OK, 1 row affected (0.00 sec)

在数据范围之内的数是可以成功插入的,但是如果超过这个范围呢

  1. mysql> insert into t1 values(128);
  2. ERROR 1264 (22003): Out of range value for column 'num' at row 1
  3. mysql> insert into t1 values(-129);
  4. ERROR 1264 (22003): Out of range value for column 'num' at row 1

显然插入失败了,mysql不像c语言在超出数据范围后会发生截断,而是直接报错,这样会倒逼程序员必须插入正确的数据。所以MySQL中的数据一定是合法的,所以数据类型就是一种约束,关于约束的话题,我们下一章再说

  • 创建无符号tinytin
  1. mysql> create table t2(num tinyint unsigned);
  2. Query OK, 0 rows affected (0.02 sec)
  3. mysql> insert into t2 values(128);
  4. Query OK, 1 row affected (0.01 sec)
  5. mysql> insert into t2 values(255);
  6. Query OK, 1 row affected (0.00 sec)
  7. mysql> select * from t2;
  8. +------+
  9. | num |
  10. +------+
  11. | 128 |
  12. | 255 |
  13. +------+
  14. 2 rows in set (0.00 sec)

2.2 bit

  • bit[(M)] : 位字段类型。M表示每个值的位数,范围从1到64。如果M被忽略,默认为1。
  1. mysql> create table t3( num1 bit(1), num2 bit(64) );
  2. Query OK, 0 rows affected (0.02 sec)
  3. mysql> insert into t3 values(0, 0);
  4. Query OK, 1 row affected (0.01 sec)
  5. mysql> insert into t3 values(1, 1);
  6. Query OK, 1 row affected (0.00 sec)
  7. mysql> insert into t3 values(2, 2);
  8. ERROR 1406 (22001): Data too long for column 'num1' at row 1
  9. mysql> select * from t3;
  10. +------+----------+
  11. | num1 | num2 |
  12. +------+----------+
  13. | | |
  14. | | |
  15. +------+----------+
  16. 2 rows in set (0.00 sec)

  • 需要注意几点:
  • 1.bit位数决定了你最大插入的值,对于num1来说,只有一位,所以只能插入0和1
  • 2.位类型现实的是ASCII值,显示不出来,如果想看,可以转换成十六进制打印
mysql> select hex(num1), hex(num2) from t3;

  1. mysql> insert into t3 values(0, 97);
  2. Query OK, 1 row affected (0.00 sec)
  3. mysql> insert into t3 values(0, 'a');
  4. Query OK, 1 row affected (0.00 sec)
  5. mysql> select * from t3;
  6. +------+----------+
  7. | num1 | num2 |
  8. +------+----------+
  9. | | |
  10. | | |
  11. | | a |
  12. | | a |
  13. +------+----------+
  14. 4 rows in set (0.00 sec)

可以看到插入'a'和97是一样的,证明了确实是显示的ASCII值

2.3浮点类型

2.3.1 float

  • float[(m, d)] [unsigned] : M指定显示长度,d指定小数位数,占用空间4个字节
     
  1. mysql> create table t4(
  2. -> num float(4, 2)
  3. -> );
  4. Query OK, 0 rows affected (0.02 sec)
  5. mysql> insert into t4 values (31.14);
  6. Query OK, 1 row affected (0.01 sec)
  7. mysql> insert into t4 values (99.99);
  8. Query OK, 1 row affected (0.00 sec)
  9. mysql> insert into t4 values (-99.99);
  10. Query OK, 1 row affected (0.00 sec)
  11. mysql> select * from t4;
  12. +--------+
  13. | num |
  14. +--------+
  15. | 31.14 |
  16. | 99.99 |
  17. | -99.99 |
  18. +--------+
  19. 3 rows in set (0.00 sec)
  20. mysql>

小数:float(4,2)表示的范围是-99.99 ~ 99.99,MySQL在保存值时会进行四舍五入。

在这个范围内的数据都是正确的,但是如果超过这个范围就无法插入。

  1. mysql> insert into t4 values (99.995);
  2. ERROR 1264 (22003): Out of range value for column 'num' at row 1
  3. mysql> insert into t4 values (-99.995);
  4. ERROR 1264 (22003): Out of range value for column 'num' at row 1


2.3.2 decimal

  • decimal(m, d) [unsigned] : 定点数m指定长度,d表示小数点的位数
     

decimal(5,2) 表示的范围是 -999.99 ~ 999.99

decimal(5,2) unsigned 表示的范围 0 ~ 999.99

decimal和float很像,但是有区别:float和decimal表示的精度不一样

  1. mysql> create table t5(
  2. -> num1 float(10, 8),
  3. -> num2 decimal(10, 8)
  4. -> );
  5. Query OK, 0 rows affected (0.02 sec)
  6. mysql> insert into t5 values (99.99, 99.99);
  7. Query OK, 1 row affected (0.01 sec)
  8. mysql> insert into t5 values (12.34567890, 12.34567890);
  9. Query OK, 1 row affected (0.00 sec)

可以看到float类型的精度会丢失,而我们给decima的是什么,他存的就是什么。

建议:如果希望小数的精度高,推荐使用decimal
 

3.字符串类型

3.1 char

  • char(L): 固定长度字符串,L是可以存储的长度,单位为字符,最大长度值可以为255

char(2) 表示可以存放两个字符,可以是字母或汉字,但是不能超过2个, 最多只能是255

  1. mysql> create table t6(
  2. -> num char(2)
  3. -> );
  4. Query OK, 0 rows affected (0.01 sec)
  5. mysql> insert into t6 values('a');
  6. Query OK, 1 row affected (0.02 sec)
  7. mysql> insert into t6 values('ab');
  8. Query OK, 1 row affected (0.01 sec)
  9. mysql> insert into t6 values('abc');
  10. ERROR 1406 (22001): Data too long for column 'num' at row 1

如果我们插入的是汉字,在gbk编码中汉字占2个字节,而utf8编码中汉字占3个字节。但是在mysql中,一个字符是二/三个字节的,所以在c语言中char(一个字节)和汉字(二/三个字节)在mysql中都占一个字符。

  1. mysql> insert into t6 values('中');
  2. Query OK, 1 row affected (0.00 sec)
  3. mysql> insert into t6 values('中国');
  4. Query OK, 1 row affected (0.00 sec)
  5. mysql> insert into t6 values('中国人');
  6. ERROR 1406 (22001): Data too long for column 'num' at row 1

3.2 varchar

  • varchar(L): 可变长度字符串,L表示字符长度,最大长度65535个字节
  1. mysql> create table tt10(id int ,name varchar(6)); --表示这里可以存放6个字符
  2. mysql> insert into tt10 values(100, 'hello');
  3. mysql> insert into tt10 values(100, '我爱你,中国');
  4. mysql> select * from tt10;
  5. +------+--------------------+
  6. | id | name |
  7. +------+--------------------+
  8. | 100 | hello |
  9. | 100 | 我爱你,中国 |
  10. +------+--------------------+

关于varchar(len),len到底是多大,这个len值,和表的编码密切相关:

  • varchar长度可以指定为0到65535之间的值,但是有1 - 3 个字节用于记录数据大小,所以说有效字节数是65532。
  • 当我们的表的编码是utf8时,varchar(n)的参数n最大值是65532/3=21844[因为utf中,一个字符占用3个字节],如果编码是gbk,varchar(n)的参数n最大是65532/2=32766(因为gbk中,一个字符占用2字节)。
  1. mysql> create table tt11(name varchar(21845))charset=utf8;
  2. ERROR 1118 (42000): Row size too large. The maximum row size for the used
  3. table type, not counting BLOBs, is 65535. You have to change some columns to
  4. TEXT or BLOBs
  5. mysql> create table tt11(name varchar(21844)) charset=utf8;
  6. Query OK, 0 rows affected (0.01 sec)

这段代码验证了utf8确实是不能超过21844

  • 关于char和varchar的区别

  • 如何选择char和varchar
  • 如果数据确定长度都一样,就使用定长(char),比如:身份证,手机号,md5
  • 如果数据长度有变化,就使用变长(varchar), 比如:名字,地址,但是你要保证最长的能存的进去。定长的磁盘空间比较浪费,但是效率高。
  • 变长的磁盘空间比较节省,但是效率低。
  • 定长的意义是,直接开辟好对应的空间
  • 变长的意义是,在不超过自定义范围的情况下,用多少,开辟多少。

4.日期类型

常用的日期有如下三个:

  • date :日期 'yyyy-mm-dd' ,占用三字节
  • datetime 时间日期格式 'yyyy-mm-dd HH:ii:ss' 表示范围从 1000 到 9999 ,占用八字节
  • timestamp :时间戳,从1970年开始的 yyyy-mm-dd HH:ii:ss 格式和 datetime 完全一致,占用四字节
  1. mysql> create table t8(
  2. -> t1 date,
  3. -> t2 datetime,
  4. -> t3 timestamp
  5. -> );
  6. Query OK, 0 rows affected (0.01 sec)
  7. mysql> desc t8;
  8. +-------+-----------+------+-----+-------------------+-----------------------------+
  9. | Field | Type | Null | Key | Default | Extra |
  10. +-------+-----------+------+-----+-------------------+-----------------------------+
  11. | t1 | date | YES | | NULL | |
  12. | t2 | datetime | YES | | NULL | |
  13. | t3 | timestamp | NO | | CURRENT_TIMESTAMP | on update CURRENT_TIMESTAMP |
  14. +-------+-----------+------+-----+-------------------+-----------------------------+

timestamp时间戳,在插入数据时会自动更新,所以我们不需要添加

  1. mysql> insert into t8(t1, t2) values ('2024-03-21', '1949-10-01 08:00:00');
  2. Query OK, 1 row affected (0.00 sec)
  3. mysql> select * from t8;
  4. +------------+---------------------+---------------------+
  5. | t1 | t2 | t3 |
  6. +------------+---------------------+---------------------+
  7. | 2024-03-21 | 1949-10-01 08:00:00 | 2024-03-21 15:28:49 |
  8. +------------+---------------------+---------------------+

当我们更新t1之后会发现,t3也会更改

  1. mysql> update t8 set t1='2003-07-31';
  2. Query OK, 1 row affected (0.01 sec)
  3. Rows matched: 1 Changed: 1 Warnings: 0
  4. mysql> select * from t8;
  5. +------------+---------------------+---------------------+
  6. | t1 | t2 | t3 |
  7. +------------+---------------------+---------------------+
  8. | 2003-07-31 | 1949-10-01 08:00:00 | 2024-03-21 15:32:57 |
  9. +------------+---------------------+---------------------+

5.枚举和集合类型

语法:

enum:枚举,“单选”类型;enum('选项1','选项2','选项3',...);

该设定只是提供了若干个选项的值,最终一个单元格中,实际只存储了其中一个值;而且出于效率考虑,这些值实际存储的是“数字”,因为这些选项的每个选项值依次对应如下数字:1,2,3,....最多65535个;当我们添加枚举值时,也可以添加对应的数字编号。

set:集合,“多选”类型; set('选项值1','选项值2','选项值3', ...);

该设定只是提供了若干个选项的值,最终一个单元格中,设计可存储了其中任意多个值;而且出于效率考虑,这些值实际存储的是“数字”,因为这些选项的每个选项值依次对应如下数字:1,2,4,8,16,32,....最多64个。

说明:不建议在添加枚举值,集合值的时候采用数字的方式,因为不利于阅读。

  1. mysql> create table t9(
  2. -> username varchar(30),
  3. -> hobby set('登山', '游泳', '篮球', '武术'),
  4. -> gender enum('男','女')
  5. -> );
  6. Query OK, 0 rows affected (0.02 sec)
  7. mysql> insert into t9 values ('张三','登山','男');
  8. Query OK, 1 row affected (0.00 sec)
  9. mysql> insert into t9 values ('张三','游泳','男');
  10. Query OK, 1 row affected (0.01 sec)
  11. mysql> insert into t9 values ('张三','游泳,篮球','男');
  12. Query OK, 1 row affected (0.01 sec)
  13. mysql> insert into t9 values ('张三','游泳,篮球,武术','男');
  14. Query OK, 1 row affected (0.00 sec)

对于enum和set都是从你创建的表中选择,不能选择其他的。否则就会报错

这两种类型也可以使用数字来表示

  1. mysql> create table t9(
  2. -> username varchar(30),
  3. -> hobby set('登山', '游泳', '篮球', '武术'),
  4. -> gender enum('男','女')
  5. -> );
  6. Query OK, 0 rows affected (0.02 sec)
  7. mysql> insert into t9 values ('王五', 2, 1);
  8. Query OK, 1 row affected (0.00 sec)
  9. mysql> insert into t9 values ('王五', 3, 2);
  10. Query OK, 1 row affected (0.01 sec)
  11. mysql> select * from t9;
  12. +----------+----------------------+--------+
  13. | username | hobby | gender |
  14. +----------+----------------------+--------+
  15. | 张三 | 登山 ||
  16. | 张三 | 游泳 ||
  17. | 张三 | 游泳,篮球 ||
  18. | 张三 | 游泳,篮球,武术 ||
  19. | 李四 | 登山,游泳 ||
  20. | 李四 | 游泳 ||
  21. | 王五 | 游泳 ||
  22. | 王五 | 登山,游泳 ||
  23. +----------+----------------------+--------+
  24. 8 rows in set (0.00 sec)

set是一个位图结构(从左往右看),第一次我们插入的是2,对应二进制为01,所以是游泳,第二次是3,对应二进制位11,就是登山和游泳

enum对应的就是下标,从1开始,对应上面男就是1,女就是2

如果我们想查询爱好中有登山的同学应该怎么查

可以看到使用这种方法只能查到hobby为登山的,不能查询所有登山的同学,我们可以使用find_in_set函数,从第二个参数中找含有第一个参数的数据

  1. mysql> select * from t9 where find_in_set('登山', hobby);
  2. +----------+---------------+--------+
  3. | username | hobby | gender |
  4. +----------+---------------+--------+
  5. | 张三 | 登山 ||
  6. | 李四 | 登山,游泳 ||
  7. | 王五 | 登山,游泳 ||
  8. +----------+---------------+--------+
  9. 3 rows in set (0.00 sec)

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/不正经/article/detail/316002
推荐阅读
相关标签
  

闽ICP备14008679号