为嘛子要慎用mysql的enum字段?(来自向东的原创博客)

本文深入分析了enum与tinyint在MySQL中的使用场景与特点,详细对比了两者在PHP弱类型环境下的表现,强调了在数据库设计中选择合适的数据类型的重要性,特别关注了enum类型在存储与检索数据时的优势与潜在混淆问题,同时讨论了tinyint作为替代方案的适用性,以及在具体实例中的实际操作与注意事项。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

背景:时下都流行enum类型的使用tinyint,那enum就真没有用的价值了么?
PHP低级编程的兄弟是这样来看这个问题的,我作下笔录如下,期望能客观的理解这个enum字段的优点及缺点:
膘哥观点:
    enum有优点。但个人觉得。。。缺点更多,客观的讲:优点主要是在建数据 库的时候就可以把一些值给规范好。缺点是。。enum不适合PHP。主要是PHP是弱类型,如:你insert into ..... set a= 1,你没法知道你是想 a= '1' 还是 a= 1(a='1'是插入值1,a=1是插入enum的第一个值,尤其php弱类型的,如果int的,很少有人在sql里加双引号。),这是PHP和mysql 在使用enum 最大的问题。所以。。安心点啦。干脆点直接tinyint。

单曲观点:
    我觉得没什么优点,对数字型的enum,简直就是梦魇,boolean  tinyint(1) 0,1 status  tinyint(1)  1,2,3,4,5,6..tinyint欢淫你~~。如:audit_result enum(1,2,3),set audit_result = 1;...容易出现膘哥所说的混淆。

简单观点:
   少用,一般都是用tinyint替代。

天枫观点:
   我觉得除了状态直观  没什么优点,我一般直接int,tinyint([1or2or3]) 到底有啥区别?(后面会简单探讨下,这里面的1or2or3区别。)

中庸观点:
   a=1是插入enum的第一个值,尤其php弱类型的,如果int的,很少有人在sql里加双引号,基本上是不加引号的。

竖琴螺观点:
   六种情况就:tinyint(1)  -1,-2,1,2,3,4 

上面各种观点重点集中在PHP这种弱类型语言对引号不重视,
程序员不写容易引起插入的语句不是自己想要的结果的问题,
容易出现int时没有用引号导致插入了新值而不是定的那个值:
  1. 表结构如下:  
  2. CREATE TABLE `enum2tinyint` (  
  3.   `switchs` enum('none','success','fail','delete','skip','1') DEFAULT NULL,  
  4.   `switch` tinyint(1) NOT NULL,  
  5.   KEY `switchs` (`switchs`)  
  6. ) ENGINE=InnoDB DEFAULT CHARSET=utf8  
  7.   
  8. 正常带引号插入enum:  
  9. INSERT INTO `test`.`enum2tinyint` (`switchs`, `switch`) VALUES ('1''1');  
  10. 1 1  
  11. PHP的弱类型问题,特别是对int类型的情况,实践如下:  
  12. 如果没有带引号插入enum字段后如下(是第一个值none):  
  13. INSERT INTO `test`.`enum2tinyint` (`switchs`, `switch`) VALUES (1, '1');  
  14. none 1  
  15. 如果没有带引号插入enum字段后如下(是第2个值success):  
  16. INSERT INTO `test`.`enum2tinyint` (`switchs`, `switch`) VALUES (2, '1');  
  17. success 1  


结论:要插入enum的值,字段必须加引号,不加引号当然是数字,数字就是key,不是value。
规劝:
1)enum是整型这样的错误很容易发生,尤其是php弱类型的,一般新来一个人,没注意enum类型,就会犯错。
2)数据库说明清楚的话,或者可选择的全是字符串的话,还没什么,但是里面有数字,难免有新手犯错,养成加引号的习惯很重要。
最终结论:
历史原因,要把enum改成tinyint程序改动太大了,用了的没必要改·~,以后新建的时候,尽量使用tinyint就好。
这种字段的重复内容过多的,索引建不建,关系不大,这种在mysql叫索引的势太低,其查询效果不太好:(英文是这么翻译的:https://justwinit.cn/post/1405/)。
————————————————————————————————————————————————————————————————————
tinyint(1)和tinyint(4)一样的,都是-127到128或者0到256。unsigned属性全是正,和c语言unsigned int有点类似:
tinyint(1)   -128 ~ 127
tinyint(1) unsigned  0 ~ 255

为何表示的最大正数比最小负数的绝对值少1呢?
第一位是符号位,1表示负数
所以负数可以到128,正数只能到127
-128正127
如:0111 1111  正数  ,负数 1111 1111
提问:
Mysql里int(1)和int(11)差别很大的,mysql的int,有个属性 ,UNSIGNED ZEROFILL  后面的ZEROFILL,就是有定义的位数不够则用零补齐对齐一下(这儿可能涉及到索引的性能罢):int,  长度(M)=5, 属性=UNSIGNED ZEROFILL(无符号,用0来填充位数),00001,00002。
但这个tinyint呢,tinyint(1)和tinyint(4)一样的?
看这文章后:http://blog.youkuaiyun.com/lysygyy/article/details/5983433
我估计:mysql这块对一个字节不像int这种四个字节要进行对其前部分进行部分字节索引,如果说tinyint(1)和tinyint(4)是一样的表示范围情况下,但mysql对tinyint数字长度位数作了可设置的限定,从理论上说其存在硬盘里的字节是不一样才是合乎逻辑的,这块估计还是为了对字节进行对齐方便索引等,以提升效率为主罢,l如:ike keyword%    索引有效,如果统一对齐,索引效果会不会更好呢?我也只是猜测。
————————————————————————————————————————————————————————————————————
在网上找了下这个enum的mysql的大体实现如下:
一)enum的存储原理我仔细查看了下手册。是这样的:
在建立这个字段时,我们会给他规定一个范围比如enum('a','b','c'),这时mysql内部会建立一张hash结构的map表,类似:0000 -> a,0001 -> b,0002 -> c。
当我插入一条数据,此字段的值位a或b或c时,他存储在里面的不是这个字符,而是对应他的索引,也就是那个0000或0001或0002。
同样,enum在mysql手册上的说明:
ENUM('value1','value2',...)
1或2个字节,取决于枚举值的个数(最多65,535个值)
除非enum的个数超过了一定数量,否则他所占的存储空间也总是1字节。


二)tinyint:
类型  字节  最小值  最大值
      (带符号的/无符号的)  (带符号的/无符号的)
TINYINT  1  -128  127
他的最小存储所占空间也是1字节。


最后,Enum,既然要用它,就不必要使用什么0,1,2来代替实际的字符串了。甚至中文字符串。他并不会对数据库性能进行多余开销。因为对于它来说,你使用'0','1','2'和'张三','李四','王五'数据表所占的存储空间一样。但是考虑到我们实际应用时数据需要从db服务器回传到web app,所以在网络传输时,当然还是尽可能的传输小数据比较好。所以如果很在意这些,还是不用它好了。


mysql中关于bit,enum,tinyint三种数据类型的差别 :
http://blog.youkuaiyun.com/shadowsniper/article/details/7071004

慎用mysql的enum字段:
http://www.neatstudio.com/show-1498-1.shtml
<think>嗯,用户想了解高校二手商品交易公告的实体属性,可能是为了设计数据库或者系统。首先,我需要根据他们提供的引用资料,看看现有的系统设计是怎么处理这个部分的。引用[1]提到数据库设计要从实际需求出发,所以得先确定公告需要哪些信息。 公告的基本属性应该包括标题、内容、发布人、时间这些。然后可能需要关联商品,比如商品ID或者分类。状态属性也很重要,比如是否生效或者已下架。附件或图片的话,可能需要单独的表来存储,避免冗余。 接着是数据库设计,主键肯定是公告ID,外键可能需要用户ID商品ID,这样能关联到用户表商品表。时间戳字段用来记录创建更新时间。状态字段可以用枚举类型,比如生效、过期、下架。 在系统开发方面,引用[2]提到使用SSM框架,所以可能需要用MyBatis处理数据访问,Spring MVC做控制层。公告管理模块应该包括发布、编辑、查询下架功能。前端可能用Vue.js,方便用户操作。 还要考虑性能优化,比如索引的添加,公告表里的用户ID商品ID可能需要索引。缓存机制可以减轻数据库压力,比如用Redis缓存热门公告。安全问题,比如防止XSS攻击,需要对用户输入的内容进行过滤。 用户可能还想知道如何实现这些功能,比如具体的代码示例或者架构设计。需要确保回答覆盖实体属性、数据库设计、系统开发流程以及相关技术实现。可能还需要提醒他们注意数据一致性事务管理,比如使用Spring的事务管理来保证操作原性。 最后,生成相关问题的时候,要围绕数据库优化、功能扩展、安全措施性能提升,这些都是用户后续可能会遇到的问题。需要确保引用正确,比如引用[1][2]分别对应数据库设计系统框架部分。</think>### 高校二手商品交易公告实体属性与系统设计 #### 一、实体属性分析(数据库设计基础) 高校二手商品交易公告的核心属性可归纳为: 1. **基础信息** - 公告ID(主键): $G\_ID$ - 公告标题: $Title$ - 公告内容: $Content$ - 发布时间: $PublishTime$ - 生效状态: $Status \in \{\text{生效}, \text{过期}, \text{下架}\}$ 2. **关联属性** - 发布者ID(外键): $UserID$ - 关联商品ID(外键): $ProductID$ - 公告分类: $Category \in \{\text{系统通知}, \text{商品促销}, \text{交易规则}\}$ 3. **扩展属性** - 附件路径: $AttachmentPath$ - 优先级: $PriorityLevel$ - 生效时间段: $ValidPeriod = [StartTime, EndTime]$ #### 二、数据库设计示例 ```sql CREATE TABLE Announcement ( G_ID INT PRIMARY KEY AUTO_INCREMENT, Title VARCHAR(100) NOT NULL, Content TEXT, PublishTime DATETIME DEFAULT CURRENT_TIMESTAMP, Status ENUM('生效','过期','下架') DEFAULT '生效', UserID INT NOT NULL, ProductID INT, Category VARCHAR(20), FOREIGN KEY (UserID) REFERENCES User(UserID), FOREIGN KEY (ProductID) REFERENCES Product(ProductID) ); ``` 此设计支持公告与用户、商品的关联,符合引用[1]强调的**实际需求导向设计原则**[^1] #### 三、系统开发要点(基于SSM框架) 1. **架构分层** - **持久层(MyBatis)**: 实现公告的CRUD操作 - **服务层(Spring)**: 包含公告状态机、有效期验证等业务逻辑 - **控制层(SpringMVC)**: 处理公告发布/修改请求 2. **关键功能实现** ```java // 公告状态变更示例 public void updateAnnouncementStatus(int gid, String newStatus) { Announcement ann = announcementMapper.selectById(gid); if (ann.getValidPeriod().contains(LocalDateTime.now())) { ann.setStatus(newStatus); announcementMapper.update(ann); } } ``` 3. **前端交互(Vue.js)** - 公告列表分页查询: 使用`axios`调用RESTful API - 富文本编辑器: 集成`Quill`实现图文混排公告内容 #### 四、安全与性能优化 1. **数据安全** - 敏感词过滤: 对公告内容进行正则表达式匹配 - XSS防护: 使用`Jsoup`清理HTML标签 2. **性能优化** - 建立联合索引: `(Category, Status)`提升查询效率 - 缓存策略: 对高频访问的公告使用Redis缓存 相关问题
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值