【MySQL】数据类型

【MySQL】数据类型

🌸心有所向,日复一日,必有精进

🌸专栏:《MySQL》

🌸作者:早凉


目录

【MySQL】数据类型​​​​​​​

数据类型的分类

类型范围

数值类型

tinyint

bit类型

小数类型

float

decimal

字符串类型

char

varchar

日期类型

enum和set

[find_ in_ set函数(集合查询使用):](#find_ in_ set函数(集合查询使用):)


数据类型的分类

类型范围

类型 字节 最小值(带符号) 最大值(带符号) 最小值(无符号) 最大值(无符号)
TINYINT 1 -128 127 0 255
SMALLINT 2 -32768 32767 0 65535
MEDIUMINT 3 -8388608 8388607 0 166777215
INT 4 -2147483648 2147483647 0 4294967295
BIGINT 8 -9223372036854775808 9223372036854775807 0 18446744073709551615

数值类型

tinyint

  • 在MySQL中,整型可以指定是有符号的和无符号的,默认是有符号的。
  • 可以通过UNSIGNED来说明某个字段是无符号的

在MySQL中,默认的tinyint生成的是有符号的,当数据超过范围时,直接报错;

但是在C/C++中:

发现并没有报错发生,这是不一样的,在学习的时候不能混为一谈;

在无符号类型的时候,插入负数直接报错;

尽量不使用unsigned,对于int类型可能存放不下的数据,int unsigned同样可能存放不

下,与其如此,还不如设计时,将int类型提升为bigint类型。

bit类型

sql 复制代码
bit[(M)] : 位字段类型。M表示每个值的位数,范围从1到64。如果M被忽略,默认为1。
  • bit字段在显示时,是按照ASCII码对应的值显示。
  • 在特定场景下,只用1位或几位bit位实现就可以使用。

小数类型

float

sql 复制代码
float[(m, d)] [unsigned] : M指定显示长度,d指定小数位数,占用空间4个字节

创建一个 float(4,2),表示的范围是-99.99 ~ 99.99,

sql 复制代码
mysql> create table t6(
    -> f1 float(4,2)
    -> );
Query OK, 0 rows affected (0.03 sec)
sql 复制代码
mysql> desc t6;
+-------+------------+------+-----+---------+-------+
| Field | Type       | Null | Key | Default | Extra |
+-------+------------+------+-----+---------+-------+
| f1    | float(4,2) | YES  |     | NULL    |       |
+-------+------------+------+-----+---------+-------+
1 row in set (0.00 sec)

当前设置的指定小数位数是2,当不足的时候补齐;

sql 复制代码
mysql> insert into t6 values(10.0);
Query OK, 1 row affected (0.01 sec)

mysql> select * from t6;
+-------+
| f1    |
+-------+
| 10.00 |
+-------+
1 row in set (0.00 sec)

当超过所设定的位数时,会进行四舍五入。当然四舍五入后的值必须在合法范围之中;

sql 复制代码
mysql> insert into t6 values(10.234);
Query OK, 1 row affected (0.00 sec)

mysql> insert into t6 values(10.235);
Query OK, 1 row affected (0.00 sec)

mysql> select * from t6;
+-------+
| f1    |
+-------+
| 10.00 |
| 10.23 |
| 10.23 |
+-------+
3 rows in set (0.00 sec)

如果定义的是float(4,2) unsigned 这时,因为把它指定为无符号的数,范围是 0 ~ 99.99

sql 复制代码
mysql> create table if not exists t7(
    -> f1 float(4,2) unsigned
    -> );
Query OK, 0 rows affected (0.03 sec)

当我们插入负数的时候直接报错了;

decimal

sql 复制代码
decimal(m, d) [unsigned] : 定点数m指定长度,d表示小数点的位数
  • decimal(5,2) 表示的范围是 -999.99 ~ 999.99
  • decimal(5,2) unsigned 表示的范围 0 ~ 999.99

但是我们有了float的浮点数存储,怎么还需要decimal?

这里只要是精度问题:

  • float表示的精度大约是7位。
  • decimal整数最大位数m为65。支持小数最大位数d是30。如果d被省略,默认为0.如果m被省略,默认是10
sql 复制代码
mysql>  create table t8 ( id int, salary float(10,8), salary2
    -> decimal(10,8));
Query OK, 0 rows affected (0.02 sec)

mysql> insert into t8 values(100,23.15431821, 23.15431821);
Query OK, 1 row affected (0.00 sec)

mysql> select * from t8;
+------+-------------+-------------+
| id   | salary      | salary2     |
+------+-------------+-------------+
|  100 | 23.15431786 | 23.15431821 |
+------+-------------+-------------+
1 row in set (0.00 sec)

# 发现decimal的精度更准确,因此如果我们希望某个数据表示高精度,选择decimal

虽然float设置的是8位精度,但是此时存储的已经与插入的差别比较大了,而decimal插入什么就存储什么,在精度要求比较高的场景下,是很好的选择;

字符串类型

char

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

对于gbk编码一个占用2个字节,utf8编码一个汉字占用3个字

这里所指的长度是字符长度,不是字节长度;

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

varchar

sql 复制代码
varchar(L): 可变长度字符串,L表示字符长度,最大长度65535个字节(mysql存储char类型,utf8默认是3字节,65535/3 = 21845)
sql 复制代码
mysql> create table if not exists t9(
    -> id int,
    -> name varchar(65535)
    -> );
ERROR 1074 (42000): Column length too big for column 'name' (max = 21845); use BLOB or TEXT instead

不是说最长为65535吗?怎么不能开辟,只能21845呀?

  • varchar长度可以指定为0到65535之间的值,但是有1 - 3 个字节用于记录数据大小,所以说有效字节数是65532。
  • 当我们的表的编码是utf8时,varchar(n)的参数n最大值是65532/3=21844(因为utf中,一个字符占用3个字节);
  • 如果编码是gbk,varchar(n)的参数n最大是65532/2=32766(因为gbk中,一个字符占用2字节);

char和varchar的不同

如何选择使用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 完全一致,占用四字节;
sql 复制代码
mysql> create table if not exists t9 (
    -> t1 date,
    -> t2 datetime,
    -> t3 timestamp
    -> );
Query OK, 0 rows affected (0.02 sec)

mysql> desc t9;
+-------+-----------+------+-----+-------------------+-----------------------------+
| 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.00 sec)

时间戳的数据会在修改当前数据的时候自动的更新,例如在视频评论区记录发布时间,不需要手动输入,根据发布时间自动更新;

sql 复制代码
mysql> insert into t9 (t1,t2) values ('2024-3-27','1949-10-01 08:00:00');
Query OK, 1 row affected (0.00 sec)

mysql> SELECT * from t9;
+------------+---------------------+---------------------+
| t1         | t2                  | t3                  |
+------------+---------------------+---------------------+
| 2024-03-27 | 1949-10-01 08:00:00 | 2024-03-27 09:37:21 |
+------------+---------------------+---------------------+
1 row in set (0.00 sec)

enum和set

sql 复制代码
enum:枚举,"单选"类型;
enum('选项1','选项2','选项3',...);

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

sql 复制代码
set:集合,"多选"类型;
set('选项值1','选项值2','选项值3', ...);

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

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

sql 复制代码
mysql> create table if not exists infor (
    -> username varchar(30),
    -> gender enum('男','女'),
    -> hobby set('舞蹈','画画','篮球','足球')
    -> );
Query OK, 0 rows affected (0.03 sec)

mysql> insert into infor values('爱坤','男','篮球');
Query OK, 1 row affected (0.00 sec)

mysql> insert into infor values('小张','女','舞蹈');
Query OK, 1 row affected (0.00 sec)

mysql> insert into infor values('小杨','女','舞蹈,画画');
Query OK, 1 row affected (0.00 sec)

mysql> insert into infor values('小李','男','舞蹈,篮球');
Query OK, 1 row affected (0.01 sec)

性别只能插入男或女,枚举类型提供了类型约束,只能插入枚举的类型。插入常量的下标也是可以的;

注意:枚举下标数字从1开始,代表第一个

对于hobby这个set集合,数字代表的不是下标,对应的是

对于插入1:00000---------->00001,代表代码这个舞蹈

对于插入3:00000----------->00011,代表的是舞蹈和画画

对于插入7:00000------------>001111,代表的是舞蹈,画画,篮球,足球;

相当于是位图结构

enum和set都是允许为NULL的。但是NULL和" "是不同的;

sql 复制代码
mysql> insert into infor values ('小花',1,0);
Query OK, 1 row affected (0.01 sec)

mysql> insert into infor (username) values ('小草');
Query OK, 1 row affected (0.00 sec)

mysql> select * from infor;
+----------+--------+---------------+
| username | gender | hobby         |
+----------+--------+---------------+
| 爱坤     | 男     | 篮球           |
| 小张     | 女     | 舞蹈           |
| 小杨     | 女     | 舞蹈,画画      |
| 小李     | 男     | 舞蹈,篮球      |
| 小花     | 男     |               |
| 小草     | NULL   | NULL          |
+----------+--------+---------------+
6 rows in set (0.00 sec)

如何查找呢?

sql 复制代码
mysql> select * from infor where hobby='篮球';
+----------+--------+--------+
| username | gender | hobby  |
+----------+--------+--------+
| 爱坤     | 男     | 篮球   |
+----------+--------+--------+
1 row in set (0.00 sec)

这里我们发现,查找的结果只是爱好单独是篮球的,那么在有些场景下需要查找只要爱好有篮球的一项都要查找的;

find_ in_ set函数(集合查询使用):

sql 复制代码
find_in_set(sub,str_list) 
  • 如果 sub 在 str_list 中,则返回下标;如果不在,返回0;
  • str_list 用逗号分隔的字符串。
sql 复制代码
mysql> select * from infor where find_in_set('篮球', hobby);
+----------+--------+---------------+
| username | gender | hobby         |
+----------+--------+---------------+
| 爱坤     | 男     | 篮球          |
| 小李     | 男     | 舞蹈,篮球     |
+----------+--------+---------------+
2 rows in set (0.00 sec)
相关推荐
夜泉_ly2 小时前
MySQL -安装与初识
数据库·mysql
qq_529835353 小时前
对计算机中缓存的理解和使用Redis作为缓存
数据库·redis·缓存
月光水岸New5 小时前
Ubuntu 中建的mysql数据库使用Navicat for MySQL连接不上
数据库·mysql·ubuntu
狄加山6755 小时前
数据库基础1
数据库
我爱松子鱼5 小时前
mysql之规则优化器RBO
数据库·mysql
chengooooooo6 小时前
苍穹外卖day8 地址上传 用户下单 订单支付
java·服务器·数据库
Rverdoser7 小时前
【SQL】多表查询案例
数据库·sql
Galeoto7 小时前
how to export a table in sqlite, and import into another
数据库·sqlite
人间打气筒(Ada)7 小时前
MySQL主从架构
服务器·数据库·mysql
leegong231117 小时前
学习PostgreSQL专家认证
数据库·学习·postgresql