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
相关推荐
Chloeis Syntax2 小时前
MySQL初阶学习日记(5)--- 联合查询
java·笔记·学习·mysql
前端不太难2 小时前
RN 版本升级、第三方库兼容、Android/iOS 崩溃(实战博文 — 从 0.63 升到 0.72)
android·ios·react
思成不止于此2 小时前
【MySQL 零基础入门】DQL 核心语法(三):学生表排序查询与分页查询篇
数据库·笔记·学习·mysql
听风吟丶2 小时前
微服务调用链追踪实战:用 SkyWalking 实现全链路可视化与故障溯源
数据库
计算机学姐2 小时前
基于Python的高校后勤报修系统【2026最新】
开发语言·vue.js·后端·python·mysql·django·flask
00后程序员张2 小时前
Transporter 的局限与替代路径,iOS 上传流程在多平台团队中的演进
android·ios·小程序·https·uni-app·iphone·webview
moluzhui2 小时前
mysql 修复插入零日期报错
数据库·mysql
习惯就好zz2 小时前
如何解包 Android boot.img 并检查 UART 是否启用
android·linux·dtc·3588·dts·解包·dtb
每次的天空2 小时前
Android车机开发——内存优化操作
android·学习·设计模式