MySQL 枚举类型如何定义比较好 tinyint?enum?varchar?

news2024/12/25 9:15:46

enum介绍

先来介绍一下enum类型吧。

ENUM 是一个字符串对象,其值通常选自一个允许值列表中,该列表在表创建时的列规格说明中被明确地列举。(建表的时候写到建表语句里)

虽然表面是字符串值,但其内部是数字索引,其索引值从1开始。

注意:下标并不是从 0 开始,而 0 则具有其它的意义(空)
ENUM数据类型提供以下优点:

  • 节省存储空间,MySQL ENUM使用数字索引(1,2,3,…)来表示字符串值。
  • 可读查询和输出,数字将转换回查询结果中的相应字符

实践

下面我们来建个表试一下这个enum类型。

CREATE TABLE `test_enum` (

    `id` INT ( 11 ) NOT NULL AUTO_INCREMENT COMMENT '主键ID',

    `color` ENUM('red','yellow','blue')  COMMENT '颜色',

    PRIMARY KEY ( `id` )

) ENGINE = INNODB AUTO_INCREMENT = 1 DEFAULT CHARSET = utf8mb4 COMMENT = '颜色表';

这是一个有red,blue,yellow三种颜色的颜色表。如果我们插入列表中的数据,是完全没问题的。

INSERT INTO test_enum(color) VALUES ('red'), ('yellow')

这个时候看一下确实使用的是数字索引:

SELECT color+0 FROM test_enum

在这里插入图片描述


但如果我们插入了一个不存在enum列表中值white,会怎么样呢?

INSERT INTO test_enum(color) VALUES ('red'), ('white')

这个时候MySQL会有个报错:

 Data truncated for column 'color' at row 2, Time: 0.017000s

这是因为我们的MySQL开启了严格模式。如果关掉严格模式,或者使用比较古老的版本,仅仅是发出了一个警告而已。
但你以为关闭了严格模式就真的可以插入成功了吗?
Naive!
你只是插入了一个空字符串,对此MySQL 官方还是作出了一点的解释:

「如果在 ENUM 列中插入无效值(即,允许值列表中不存在的字符串),则会插入空字符串 ( ‘’ )
作为特殊错误值,这个空字符串可以通过此字符串具有数字值 0 来区分 正常 的空字符串 」

翻译一下:

就是说如果往 enum 列中插入了无效的值,可以被插入,但插入的是一个特殊的空字符串,而该空字符串的数值是 0。所以正常的enum索引从1开始,0就是留给这个空字符串的。
在这里插入图片描述

但是0和‘0’还是不一样的

INSERT INTO test_enum(color) VALUES  (0)

这个时候会报错:

 Data truncated for column 'color' at row 1, Time: 0.001000s

如果这样,就可以:

INSERT INTO test_enum(color) VALUES  ('0')

所以enum类型对于php等弱语言类型的支持很差,弱语言类型打引号和不打引号的值可能是同一类型,但是对于mysql中enum类型的字段来说,那就不一定是一回事了。

这里可以看出来enum类型确实对枚举值进行了一个校验,但是如果我们这个字段需要新增一种颜色而表结构忘记改了,那么我们以为插入数据成功了,其实只是插入一个空,这点真是一个值得注意的坑。

如果使用数字作为ENUM枚举常量,这种双重性很容易导致混乱,例如ENUM(‘1’,’2’,’3’)。建议尽量避免这么做。

所以超级不推荐在mysql中设置某一字段类型为enum,但是存的值为数字,比如‘0’,‘1’,‘2’;

另外,对于enum的排序也是个问题,其不是按枚举值字母顺序排序的,而是按数字索引排序的,也就是按你定义的顺序排序的。这点也是需要特别注意的。

SELECT color FROM test_enum order by color


总结一下enum主要优点:

  • 数据更紧凑,节省存储空间。因为 ENUM 列一般都是有限的值,一般不多余 5 个这样,这就比保存 true 或 false
    节省空间多了。因为 MySQL 会在创建或者修改表结构时将 enum 允许的值自动编码为数字,而这个数字一般的分配空间为 1~2 字节
    ( byte ) ,具体取决于实现。 例如,将值为 yellow 的100万行插入表将需要 100 万字节的存储空间,而如果将实际字符串
    yellow存储在 VARCHAR 列中则需要 600 万字节。

  • 更好的可读性,虽然在存储的是数字,但在输入和输出时使用的都是对应的字符值。

  • 如果启用了严格的SQL模式,错误值会导致警告或错误,可在一定程度上过滤掉脏数据。

但是,他的缺点却更多:

  • 在MySQL语句中可使用ENUM的索引值,也可以使用字符串,容易误用,尤其对于数字型枚举值。
  • 更改enum枚举成员需要使用ALTER
    TABLE语句重建整个表,大部分情况下会进行全表扫描;对于一些列未来可能会改变的字符串,使用枚举不是一个好主意,除非能接受只在列表末尾添加元素。
  • 如果启用了严格的 SQL 模式 ( sql_mode ) ,尝试插入无效的 ENUM
    值会导致错误。如果关闭严格模式插入的数据仍然是错误的。
  • 字符枚举值排序是按定义顺序排的而不是按字母顺序排的,如果想按字母顺序需要 ORDER BY CONCAT(col)。
  • ENUM类型不是SQL标准,属于MySQL,而其他DBMS不一定有原生的支持。
  • 枚举值不能是表达式,即使是计算字符串值的表达式也是如此。

看了这么多缺点,我还是决定放弃使用enum了。

那么情况下可以使用enum呢?

  • 你的enum值是固定不变的,比如扑克牌的花色等。
  • enum的值数量大于2个并少于10个。(我觉得超过10个就很难管理与使用了)
  • 这个表不需要存储额外的关联信息。比如扑克牌花色想关联黑色和红色那么就很难了。

最后的建议:

非常不建议使用ENUM存数字,如果搭配弱类型语言,那简直就是给自己找麻烦。
尽量不要用这个类型,除非你非常确定你的枚举成员不会改变,还有你没有转换数据库的需求;
如果字段是字符串,并且长度固定,建议用char类型;如果不固定且没有搜索要求用varchar也没啥问题。
如果是数值型,建议使用tinyint,只占1个字节,比较稳妥。就是这个字段备注要写清楚了,不然别人看表全是数字一脸懵逼。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/974368.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Nvidia vpi2 不是最新版本问题

问题: 根据官方手册:VPI - Vision Programming Interface: Installation安装的vpi2有可能不是最新版本,主要原因是手册中的添加的源不是最新的。 解决方法: 在 Index网址中搜索 vpi,如下 可以看到,最新的…

【数字通信原理】笔记(持续更新ing)

通信原理学习笔记,课程见b站: 由于教材不同,我们的课程使用的是《数字通信原理》主编:李白萍 版本,因此此笔记以我们的教材为主整理up主的笔记。 详情见:通信原理 文章目录 第一章 绪论1. 通信的基本概念2. 信息的量度3. 通信系统的性能指标 …

Mars3d图层树//图层管理加载时设置默认折叠的状态

问题:Mars3d图层树//图层管理加载时设置默认折叠的状态。 设置参考: 有个参数 open:false,写在对应的图层配置下,比如某个节点不展示,就对这个节点配置下。 如果全局,可以搜下这个widget内的代码&#xff…

YOLO目标检测——赛马数据集下载分享

目标检测赛马数据集在马匹竞赛、马匹健康监测、马匹行为研究、马匹安全监控和马匹图像检索等应用场景中具有广泛的应用潜力,可以为马匹产业的发展和管理提供有力支持 数据集点击下载:YOLO赛马数据集640图片标框.rar 更多数据集下载和效果展示&#x…

Nature Medicine:GWAS揭示抑郁症与其他精神疾病风险之间的关联

几乎五分之一的丹麦人在一生中经历过抑郁症。奥尔胡斯大学的一项新研究现在表明,抑郁症的遗传风险与其他精神病诊断的遗传风险增加相关。这项研究于2023年7月18日发表在《Nature Medicine》(IF202282.9)杂志上。 研究人员分析了130万人&#…

高光谱图像超分辨率-总

高光谱图像超分辨率 高光谱图像超分辨率 高光谱图像超分辨率一、基础内容1.1 高光谱图像特点1.2 研究现状1.3 高光谱图像数据集1.4 评价指标1.5 Wald**协议**二、文献阅读清单2.1 综述+先锋工作1.提出解混的思想。2.随机混合模型在高光谱分辨率增强中的应用。3.遥感中的多光谱和…

MySQL基础(六)流程控制、游标、触发器、窗口函数

目录 定义条件与处理程序 定义条件 定义处理程序 流程控制 IF 分支结构之 CASE 循环结构之LOOP 循环结构之WHILE 循环结构之REPEAT 跳转语句leave和iterate 游标 使用游标步骤 全局变量的持久化 触发器 触发器的使用 查看触发器 删除触发器 定义条件与处理程序 …

基于 LLM 的知识图谱另类实践

本文整理自社区用户陈卓见在「夜谈 LLM」主题分享上的演讲,主要包括以下内容: 利用大模型构建知识图谱利用大模型操作结构化数据利用大模型使用工具 利用大模型构建知识图谱 上图是之前,我基于大语言模型构建知识图谱的成品图,主…

【Java从0到1学习】14 Java多线程

1. 多线程概述 人们在日常生活中,很多事情都是可以同时进行的。例如,一个人可以一边听音乐,一边打扫房间,可以一边吃饭,一边看电视。在使用计算机时,很多任务也是可以同时进行的。例如,可以一边…

Android逆向学习(番外一)smali2java部分文件无法反编译的bug与修复方法

Android逆向学习(番外一)smali2java部分文件无法反编译的bug与修复方法 一、前言 昨天我和往常一样准备着android逆向(四)的博客,结果发现smali2java对某些文件无法进行逆向,我不知道windows会不会产生这…

安全运营中心(SOC)技术框架

2018年曾经画过一个安全运营体系框架,基本思路是在基础单点技术防护体系基础上,围绕着动态防御、深度分析、实时检测,建立安全运营大数据分析平台,可以算作是解决方案产品的思路。 依据这个体系框架,当时写了《基于主动…

C++零碎记录(三)

作者:小王同学在积累 链接:https://www.zhihu.com/question/437657370/answer/1692846096 来源:知乎 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。 5. 构造函数和析构函数 5.1 构造函数和析构函数的…

史上最全的计算机发展编年史!!!

今天分享给大家,以供参考。 1614年 苏格兰人约翰纳皮尔(John Napier)发表了一篇论文,其中提到他发明了一种可以计算四则运算和方根运算的精巧装置。 1623年 威廉契克卡德(Wilhelm Schickard)制作了一个通过…

【从0学习Solidity】2. 值类型详解

Solidity极简入门: 2. 值类型 博主简介:不写代码没饭吃,一名全栈领域的创作者,专注于研究互联网产品的解决方案和技术。熟悉云原生、微服务架构,分享一些项目实战经验以及前沿技术的见解。关注我们的主页,探索全栈开发…

计算机网络-谢希任第八版学习笔记总结

一.计算机网络概述 21世纪三个特点 数字化 信息化 智能化,其中主要是围绕智能化。 网络的常见分类: 电话网络 有线电视网络 计算机网络 互联网:Internet 由数量极大的计算机网络相连接 特点: 共享性 连通性 互联网&…

MySql学习笔记09——索引介绍

索引 基本概念 通过给字段名上增加索引,可以提高查询的效率。 一张表的一个字段可以添加一个索引,当然,多个字段联合起来也可以添加索引。索引相当于一本书的目录,是为了缩小扫描范围而存在的一种机制。 select * from t_user…

Java“牵手”阿里巴巴商品列表数据,关键词搜索阿里巴巴商品数据接口,阿里巴巴API申请指南

阿里巴巴商城是一个网上购物平台,售卖各类商品,包括服装、鞋类、家居用品、美妆产品、电子产品等。要获取阿里巴巴商品列表和商品详情页面数据,您可以通过开放平台的接口或者直接访问阿里巴巴商城的网页来获取商品详情信息。以下是两种常用方…

纯前端实现 导入 与 导出 Excel

最近经常在做 不规则Excel的导入,或者一些普通Excel的导出,当前以上说的都是纯前端来实现;下面我们来聊聊经常用到的Excel导出与导入的实现方案,本文实现技术栈以 Vue2 JS 为例 导入分类: 调用 API 完全由后端来解析数…

学习心得07:C#

之前也没有看过C#的书,C#的程序倒是搞了一些。好在项目不大,我又会套路。 C#很象是JAVA。好像就是JAVA出来之后,微软抄的。好东西就要学习,这不丢脸。 我倒是想,有没有办法把JAVA和C#进行映射,然后直接编译…

优思学院|亲和图案例:寻找六西格玛的项目

什么是亲和图? 亲和图(Affinity Diagram)主要功能在於分类归纳,协助在一堆杂乱无章的资料之中,有系统的归纳出几个大类,以利后续作业。通常先利用头脑风暴(Brainstorming)方式得到大…