<?xml version="1.0" encoding="UTF-8" ?>
<rss version="2.0">
<channel>
<title><![CDATA[向东博客 专注WEB应用 构架之美 --- 构架之美，在于尽态极妍 | 应用之美，在于药到病除]]></title> 
<link>http://www.jackxiang.com/index.php</link> 
<description><![CDATA[赢在IT，Playin' with IT,Focus on Killer Application,Marketing Meets Technology.]]></description> 
<language>zh-cn</language> 
<copyright><![CDATA[向东博客 专注WEB应用 构架之美 --- 构架之美，在于尽态极妍 | 应用之美，在于药到病除]]></copyright>
<item>
<link>http://www.jackxiang.com/post//</link>
<title><![CDATA[为嘛子要慎用mysql的enum字段？]]></title> 
<author>jack &lt;xdy108@126.com&gt;</author>
<category><![CDATA[Php/Js/Shell/Go]]></category>
<pubDate>Wed, 30 Jul 2014 08:17:20 +0000</pubDate> 
<guid>http://www.jackxiang.com/post//</guid> 
<description>
<![CDATA[ 
	背景：时下都流行enum类型的使用tinyint，那enum就真没有用的价值了么？<br/>PHP低级编程的兄弟是这样来看这个问题的，我作下笔录如下，期望能客观的理解这个enum字段的优点及缺点：<br/>膘哥观点：<br/>&nbsp;&nbsp;&nbsp;&nbsp;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。<br/><br/>单曲观点：<br/>&nbsp;&nbsp;&nbsp;&nbsp;我觉得没什么优点，对数字型的enum，简直就是梦魇，boolean&nbsp;&nbsp;tinyint(1) 0,1 status&nbsp;&nbsp;tinyint(1)&nbsp;&nbsp;1,2,3,4,5,6..tinyint欢淫你～～。如：audit_result enum(1,2,3)，set audit_result = 1;...容易出现膘哥所说的混淆。<br/><br/>简单观点：<br/>&nbsp;&nbsp; 少用，一般都是用tinyint替代。<br/><br/>天枫观点：<br/>&nbsp;&nbsp; 我觉得除了状态直观&nbsp;&nbsp;没什么优点，我一般直接int，tinyint（[1or2or3]） 到底有啥区别？（后面会简单探讨下，这里面的1or2or3区别。）<br/><br/>中庸观点：<br/>&nbsp;&nbsp; a=1是插入enum的第一个值，尤其php弱类型的，如果int的，很少有人在sql里加双引号，基本上是不加引号的。<br/><br/>竖琴螺观点：<br/>&nbsp;&nbsp; 六种情况就：tinyint(1)&nbsp;&nbsp;-1，-2，1,2,3,4 <br/><br/>上面各种观点重点集中在PHP这种弱类型语言对引号不重视，<br/>程序员不写容易引起插入的语句不是自己想要的结果的问题，<br/>容易出现int时没有用引号导致插入了新值而不是定的那个值：<br/><textarea name="code" class="C" rows="15" cols="100">
表结构如下：
CREATE TABLE `enum2tinyint` (
&nbsp;&nbsp;`switchs` enum('none','success','fail','delete','skip','1') DEFAULT NULL,
&nbsp;&nbsp;`switch` tinyint(1) NOT NULL,
&nbsp;&nbsp;KEY `switchs` (`switchs`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8

正常带引号插入enum：
INSERT INTO `test`.`enum2tinyint` (`switchs`, `switch`) VALUES ('1', '1');
1 1
PHP的弱类型问题，特别是对int类型的情况，实践如下：
如果没有带引号插入enum字段后如下（是第一个值none）：
INSERT INTO `test`.`enum2tinyint` (`switchs`, `switch`) VALUES (1, '1');
none 1
如果没有带引号插入enum字段后如下（是第2个值success）：
INSERT INTO `test`.`enum2tinyint` (`switchs`, `switch`) VALUES (2, '1');
success 1
</textarea><br/><br/>结论：要插入enum的值，字段必须加引号，不加引号当然是数字，数字就是key，不是value。<br/>规劝：<br/>1）enum是整型这样的错误很容易发生，尤其是php弱类型的，一般新来一个人，没注意enum类型，就会犯错。<br/>2）数据库说明清楚的话，或者可选择的全是字符串的话，还没什么，但是里面有数字，难免有新手犯错，养成加引号的习惯很重要。<br/>最终结论：<br/>历史原因，要把enum改成tinyint程序改动太大了，用了的没必要改·～，以后新建的时候，尽量使用tinyint就好。<br/>这种字段的重复内容过多的，索引建不建，关系不大，这种在mysql叫索引的势太低，其查询效果不太好：（英文是这么翻译的：https://jackxiang.com/post/1405/）。<br/>————————————————————————————————————————————————————————————————————<br/>tinyint(1)和tinyint(4)一样的，都是-127到128或者0到256。unsigned属性全是正，和c语言unsigned int有点类似：<br/>tinyint(1)&nbsp;&nbsp; -128 ~ 127<br/>tinyint(1) unsigned&nbsp;&nbsp;0 ~ 255<br/><br/>为何表示的最大正数比最小负数的绝对值少1呢？<br/>第一位是符号位，1表示负数<br/>所以负数可以到128,正数只能到127<br/>-128正127<br/>如：0111 1111&nbsp;&nbsp;正数&nbsp;&nbsp;，负数 1111 1111<br/>提问：<br/>Mysql里int(1)和int(11)差别很大的，mysql的int,有个属性 ，UNSIGNED ZEROFILL&nbsp;&nbsp;后面的ZEROFILL，就是有定义的位数不够则用零补齐对齐一下（这儿可能涉及到索引的性能罢）：int,&nbsp;&nbsp;长度(M)=5, 属性=UNSIGNED ZEROFILL(无符号,用0来填充位数),00001,00002。<br/>但这个tinyint呢，tinyint(1)和tinyint(4)一样的？<br/>看这文章后：http://blog.csdn.net/lysygyy/article/details/5983433<br/>我估计：mysql这块对一个字节不像int这种四个字节要进行对其前部分进行部分字节索引，如果说tinyint(1)和tinyint(4)是一样的表示范围情况下，但mysql对tinyint数字长度位数作了可设置的限定，从理论上说其存在硬盘里的字节是不一样才是合乎逻辑的，这块估计还是为了对字节进行对齐方便索引等，以提升效率为主罢，l如：ike keyword%&nbsp;&nbsp;&nbsp;&nbsp;索引有效，如果统一对齐，索引效果会不会更好呢？我也只是猜测。<br/>————————————————————————————————————————————————————————————————————<br/>在网上找了下这个enum的mysql的大体实现如下：<br/>一）enum的存储原理我仔细查看了下手册。是这样的:<br/>在建立这个字段时，我们会给他规定一个范围比如enum('a','b','c'),这时mysql内部会建立一张hash结构的map表，类似:0000 -> a，0001 -> b，0002 -> c。<br/>当我插入一条数据，此字段的值位a或b或c时，他存储在里面的不是这个字符，而是对应他的索引，也就是那个0000或0001或0002。<br/>同样，enum在mysql手册上的说明:<br/>ENUM('value1','value2',...)<br/>1或2个字节，取决于枚举值的个数(最多65,535个值)<br/>除非enum的个数超过了一定数量，否则他所占的存储空间也总是1字节。<br/><br/><br/>二）tinyint:<br/>类型&nbsp;&nbsp;字节&nbsp;&nbsp;最小值&nbsp;&nbsp;最大值<br/>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;(带符号的/无符号的)&nbsp;&nbsp;(带符号的/无符号的)<br/>TINYINT&nbsp;&nbsp;1&nbsp;&nbsp;-128&nbsp;&nbsp;127<br/>他的最小存储所占空间也是1字节。<br/><br/><br/>最后，Enum，既然要用它，就不必要使用什么0，1，2来代替实际的字符串了。甚至中文字符串。他并不会对数据库性能进行多余开销。因为对于它来说，你使用'0','1','2'和'张三','李四','王五'数据表所占的存储空间一样。但是考虑到我们实际应用时数据需要从db服务器回传到web app，所以在网络传输时，当然还是尽可能的传输小数据比较好。所以如果很在意这些，还是不用它好了。<br/><br/><br/>mysql中关于bit,enum,tinyint三种数据类型的差别 ：<br/>http://blog.csdn.net/shadowsniper/article/details/7071004<br/><br/>慎用mysql的enum字段：<br/>http://www.neatstudio.com/show-1498-1.shtml
]]>
</description>
</item><item>
<link>http://www.jackxiang.com/post//#blogcomment</link>
<title><![CDATA[[评论] 为嘛子要慎用mysql的enum字段？]]></title> 
<author> &lt;user@domain.com&gt;</author>
<category><![CDATA[评论]]></category>
<pubDate>Thu, 01 Jan 1970 00:00:00 +0000</pubDate> 
<guid>http://www.jackxiang.com/post//#blogcomment</guid> 
<description>
<![CDATA[ 
	
]]>
</description>
</item>
</channel>
</rss>