mysql 类型自动化转换问题
mysql 类型自动化转换问题
背景
有个业务需求,使用到find_in_set函数,简单贴下,如下:
SELECT FIND_IN_SET('b','a,b,c,d'); //返回值为2,即第2个值
实际用法:
select * from campaign_plan where find_in_set('4',ad_type);
参考:https://www.cnblogs.com/mytzq/p/7090197.html
无意间发现了一个这样的问题,ad _type为text类型,具体表数据如下:
执行了语句如下:
顿时三观崩了,什么情况,怎么回匹配到第一个数字呢?
解惑
询问了一个朋友,他给了一个方向,mysql会自动转换类型,当你select语句的条件和数据中的type不一致的时候;
一拍大腿,是呀,之前有碰到过类似问题,如下:
有个id是varchar,你不加‘’也可以查询出来, 应该是转换了的,但这样没走索引的,所以查询效率很低 。。
当时碰到的问题是,当数据是id='123', 你select是where id = 123,一个是int 一个是varchar,mysql会自动的'123'转为123跟你的对比
回到,我上面碰到的问题也是类似,只是有一个逗号的存在,在转换的时候会截断,实际操作如下:
所以就解释通了,三观恢复
拿了一下官方的翻译版本,
隐式类型转换规则:
- 如果一个或两个参数都是NULL,比较的结果是NULL,除了NULL安全的<=>相等比较运算符。对于NULL <=> NULL,结果为true。不需要转换
- 如果比较操作中的两个参数都是字符串,则将它们作为字符串进行比较。
- 如果两个参数都是整数,则将它们作为整数进行比较。
- 如果不与数字进行比较,则将十六进制值视为二进制字符串
- 如果其中一个参数是十进制值,则比较取决于另一个参数。 如果另一个参数是十进制或整数值,则将参数与十进制值进行比较,如果另一个参数是浮点值,则将参数与浮点值进行比较
- 如果其中一个参数是TIMESTAMP或DATETIME列,另一个参数是常量,则在执行比较之前将常量转换为时间戳。
- 在所有其他情况下,参数都是作为浮点数(实数)比较的。
实际例子:
mysql> SELECT 1+'1';
-> 2
mysql> SELECT CONCAT(2,' test');
-> '2 test'
mysql> SELECT 1 > '6x';
-> 0
mysql> SELECT 7 > '6x';
-> 1
mysql> SELECT 0 > 'x6';
-> 0
mysql> SELECT 0 = 'x6';
-> 1
参考资料
https://zhuanlan.zhihu.com/p/30955365
https://dev.mysql.com/doc/refman/5.5/en/type-conversion.html