MySQL 数据类型

目录

[一、数值类型:别让数据 "越界"](#一、数值类型:别让数据 “越界”)

[1. 整数类型:tinyint/int 的坑](#1. 整数类型:tinyint/int 的坑)

[2. 小数类型:float vs decimal,别再用错了!](#2. 小数类型:float vs decimal,别再用错了!)

[二、字符串类型:char 和 varchar 怎么选?](#二、字符串类型:char 和 varchar 怎么选?)

[三、日期类型:datetime 和 timestamp 的 "自动更新" 坑](#三、日期类型:datetime 和 timestamp 的 “自动更新” 坑)

[四、枚举类型:enum/set 的 "多选查询" 坑](#四、枚举类型:enum/set 的 “多选查询” 坑)

总结:选型口诀


作为后端开发,MySQL 数据类型选不对,轻则浪费存储,重则触发 "离奇" bug。这篇结合实战案例,拆解常用数据类型的用法与坑点~

一、数值类型:别让数据 "越界"

数值类型最容易踩的坑是范围溢出,先看几个核心类型:

1. 整数类型:tinyint/int 的坑

tinyint为例:

  • 默认有符号 ,范围是 -128 ~ 127
  • unsigned后范围变为 0 ~ 255

如果硬插超范围值,直接报错:

sql 复制代码
create table tt1(num tinyint);
insert into tt1 values(128); -- 报错:out of range

建议:尽量不用unsigned,int 存不下就升级为bigint,更稳妥。

2. 小数类型:float vs decimal,别再用错了!

float 是 "近似值",decimal 是 "精确值",看案例:

sql 复制代码
create table tt8(
  id int,
  salary float(10,8),   -- 近似存储
  salary2 decimal(10,8) -- 精确存储
);
insert into tt8 values(100,23.12345612,23.12345612);
select * from tt8;
-- 结果:salary显示23.12345695(丢精度),salary2显示23.12345612(精确)

结论 :存金额、税率等用decimal,存非高精度值(如身高)用float

二、字符串类型:char 和 varchar 怎么选?

这俩都是存字符串,但适用场景天差地别:

特性 char(L) varchar(L)
长度 固定(L 为字符数) 可变(L 为最大字符数)
存储开销 浪费空间,但查询快 省空间,但查询略慢
适用场景 固定长度数据(身份证) 可变长度数据(姓名)

案例:char(2)能存 2 个字符(字母 / 汉字),varchar(6)可存最多 6 个字符:

sql 复制代码
create table tt9(id int, name char(2));
insert into tt9 values(101, '中国'); -- 正常(2个汉字)

create table tt10(id int, name varchar(6));
insert into tt10 values(100, '我爱你,中国'); -- 正常(6个字符)

三、日期类型:datetime 和 timestamp 的 "自动更新" 坑

常用日期类型有 3 个,但timestamp有个隐藏特性:自动更新时间

看案例:

sql 复制代码
create table birthday(t1 date, t2 datetime, t3 timestamp);
insert into birthday(t1,t2) values('1997-7-1','2008-8-8 12:1:1');
select * from birthday;
-- t3自动填充"插入时的当前时间"

update birthday set t1='2000-1-1';
select * from birthday;
-- t3自动更新为"修改时的当前时间"

结论

  • 存 "创建 / 更新时间" 用timestamp(自动维护);
  • 存固定时间(如订单支付时间)用datetime

四、枚举类型:enum/set 的 "多选查询" 坑

如果字段是固定选项(如性别、爱好),可用enum(单选)或set(多选),但set查询要注意:不能直接用=匹配

案例:存 "爱好" 用set,查询 "喜欢登山的人":

cpp 复制代码
create table votes(
  username varchar(30),
  hobby set('登山','游泳','篮球','武术'), -- 多选
  gender enum('男','女')
);
insert into votes values('雷锋','登山,武术','男');

-- 错误写法:查不到"登山+武术"的记录
select * from votes where hobby='登山';

-- 正确写法:用find_in_set函数
select * from votes where find_in_set('登山', hobby);

总结:选型口诀

  1. 整数选int,海量用bigint,别用unsigned
  2. 小数精度选decimal,非精度选float
  3. 定长字符用char,变长用varchar
  4. 自动时间戳用timestamp,固定时间用datetime
  5. 多选查询set字段,记得用find_in_set
相关推荐
剩下了什么3 小时前
MySQL JSON_SET() 函数
数据库·mysql·json
山峰哥3 小时前
数据库工程与SQL调优——从索引策略到查询优化的深度实践
数据库·sql·性能优化·编辑器
较劲男子汉4 小时前
CANN Runtime零拷贝传输技术源码实战 彻底打通Host与Device的数据传输壁垒
运维·服务器·数据库·cann
java搬砖工-苤-初心不变4 小时前
MySQL 主从复制配置完全指南:从原理到实践
数据库·mysql
Doro再努力4 小时前
【Linux操作系统10】Makefile深度解析:从依赖推导到有效编译
android·linux·运维·服务器·编辑器·vim
Daniel李华4 小时前
echarts使用案例
android·javascript·echarts
WangYaolove13145 小时前
基于python的在线水果销售系统(源码+文档)
python·mysql·django·毕业设计·源码
做人不要太理性5 小时前
CANN Runtime 运行时组件深度解析:任务调度机制、存储管理策略与维测体系构建逻辑
android·运维·魔珐星云
山岚的运维笔记5 小时前
SQL Server笔记 -- 第18章:Views
数据库·笔记·sql·microsoft·sqlserver
我命由我123456 小时前
Android 广播 - 静态注册与动态注册对广播接收器实例创建的影响
android·java·开发语言·java-ee·android studio·android-studio·android runtime