设为首页 收藏本站
查看: 252|回复: 0

[经验分享] MySQL中隐式转换

[复制链接]
累计签到:1 天
连续签到:1 天
发表于 2014-9-15 09:39:13 | 显示全部楼层 |阅读模式
    MySQL [test]> show create table emp\G  
    *************************** 1. row ***************************  
           Table: emp  
    Create Table: CREATE TABLE `emp` (  
      `EMPNO` int(11) NOT NULL,  
      `ENAME` varchar(15) NOT NULL,  
      `JOB` varchar(15) NOT NULL,  
      `MGR` int(11) DEFAULT '0',  
      `HIREDATE` timestamp NULL DEFAULT NULL,  
      `SAL` int(20) DEFAULT '0',  
      `COMM` int(11) DEFAULT '0',  
      `DEPTNO` int(11) NOT NULL,  
      PRIMARY KEY (`EMPNO`),  
      KEY `idx_deptno` (`DEPTNO`),  
      KEY `idx_sal` (`SAL`),  
      KEY `idx_comm` (`COMM`),  
      KEY `idx_ename` (`ENAME`)  
    ) ENGINE=InnoDB DEFAULT CHARSET=utf8  
    1 row in set (0.00 sec)  



    </pre><pre name="code" class="sql">MySQL [test]> select * from emp;  
    +-------+--------+-----------+------+---------------------+------+------+--------+  
    | EMPNO | ENAME  | JOB       | MGR  | HIREDATE            | SAL  | COMM | DEPTNO |  
    +-------+--------+-----------+------+---------------------+------+------+--------+  
    |  7369 | SMITH  | CLERK     | 7902 | 1980-12-17 00:00:00 |  800 | NULL |     20 |  
    |  7499 | ALLEN  | SALESMAN  | 7698 | 1981-02-20 00:00:00 | 1600 |  300 |     30 |  
    |  7521 | WARD   | SALESMAN  | 7698 | 1981-02-22 00:00:00 | 1250 |  500 |     30 |  
    |  7566 | JONES  | MANAGER   | 7839 | 1981-04-02 00:00:00 | 2975 |    0 |     20 |  
    |  7654 | MARTIN | SALESMAN  | 7698 | 1981-09-28 00:00:00 | 1250 | 1400 |     30 |  
    |  7698 | BLAKE  | MANAGER   | 7839 | 1981-05-01 00:00:00 | 2850 |    0 |     30 |  
    |  7782 | CLARK  | MANAGER   | 7839 | 1981-06-09 00:00:00 | 2450 |    0 |     10 |  
    |  7788 | SCOTT  | ANALYST   | 7566 | 1987-04-19 00:00:00 | 3000 | NULL |     20 |  
    |  7839 | KING   | PRESIDENT |    0 | 1981-11-17 00:00:00 | 5000 |    0 |     10 |  
    |  7844 | TURNER | SALESMAN  | 7698 | 1981-09-08 00:00:00 | 1500 |    0 |     30 |  
    |  7876 | ADAMS  | CLERK     | 7788 | 1987-05-23 00:00:00 | 1100 |    0 |     20 |  
    |  7900 | JAMES  | CLERK     | 7698 | 1981-12-03 00:00:00 |  950 |    0 |     30 |  
    |  7902 | FORD   | ANALYST   | 7566 | 1981-12-03 00:00:00 | 3000 |    0 |     20 |  
    |  7934 | MILLER | CLERK     | 7782 | 1982-01-23 00:00:00 | 1300 |    0 |     10 |  
    |  7936 | 23456  | BOSCO-DBA | 7788 | 2014-09-13 16:13:56 | 2450 |  800 |     10 |  
    +-------+--------+-----------+------+---------------------+------+------+--------+  
    15 rows in set (0.00 sec)  


2. 数值类型(int)

首先提个问题,如上测试表emp中empno是主键,类型为int,那么:



    select * from emp where empno='7788';  

会产生隐式转换吗?


下面实验证明:



    MySQL [test]> select * from emp where empno=7788;  
    +-------+-------+---------+------+---------------------+------+------+--------+  
    | EMPNO | ENAME | JOB     | MGR  | HIREDATE            | SAL  | COMM | DEPTNO |  
    +-------+-------+---------+------+---------------------+------+------+--------+  
    |  7788 | SCOTT | ANALYST | 7566 | 1987-04-19 00:00:00 | 3000 | NULL |     20 |  
    +-------+-------+---------+------+---------------------+------+------+--------+  
    1 row in set (0.00 sec)  
      
    MySQL [test]> explain select * from emp where empno=7788;  
    +----+-------------+-------+-------+---------------+---------+---------+-------+------+-------+  
    | id | select_type | table | type  | possible_keys | key     | key_len | ref   | rows | Extra |  
    +----+-------------+-------+-------+---------------+---------+---------+-------+------+-------+  
    |  1 | SIMPLE      | emp   | const | PRIMARY       | PRIMARY | 4       | const |    1 | NULL  |  
    +----+-------------+-------+-------+---------------+---------+---------+-------+------+-------+  
    1 row in set (0.00 sec)  
      
    MySQL [test]> select * from emp where empno='7788';  
    +-------+-------+---------+------+---------------------+------+------+--------+  
    | EMPNO | ENAME | JOB     | MGR  | HIREDATE            | SAL  | COMM | DEPTNO |  
    +-------+-------+---------+------+---------------------+------+------+--------+  
    |  7788 | SCOTT | ANALYST | 7566 | 1987-04-19 00:00:00 | 3000 | NULL |     20 |  
    +-------+-------+---------+------+---------------------+------+------+--------+  
    1 row in set (0.00 sec)  
      
    MySQL [test]> explain select * from emp where empno='7788';  
    +----+-------------+-------+-------+---------------+---------+---------+-------+------+-------+  
    | id | select_type | table | type  | possible_keys | key     | key_len | ref   | rows | Extra |  
    +----+-------------+-------+-------+---------------+---------+---------+-------+------+-------+  
    |  1 | SIMPLE      | emp   | const | PRIMARY       | PRIMARY | 4       | const |    1 | NULL  |  
    +----+-------------+-------+-------+---------------+---------+---------+-------+------+-------+  
    1 row in set (0.00 sec)  


可见,针对数据类型字段,即使类型不一致,并不影响是否使用索引,执行计划是一样的,不会产生隐式转换。但仍然建议在生产库中尽量避免出现这样的SQL。


注意:

数值类型有一种隐式转换,如果以数字开关的,后面的字符将被截断,只取前面的数字值,如果不以数字开关的将被置为0。如下:



    MySQL [test]> select * from emp where empno='7788ab12';   ## 这个就相当于empno=7788,后面的ab12将被截断,并且不影响索引的使用  
    +-------+-------+---------+------+---------------------+------+------+--------+  
    | EMPNO | ENAME | JOB     | MGR  | HIREDATE            | SAL  | COMM | DEPTNO |  
    +-------+-------+---------+------+---------------------+------+------+--------+  
    |  7788 | SCOTT | ANALYST | 7566 | 1987-04-19 00:00:00 | 3000 | NULL |     20 |  
    +-------+-------+---------+------+---------------------+------+------+--------+  
    1 row in set, 1 warning (0.00 sec)  
      
    MySQL [test]> show warnings;  
    +---------+------+----------------------------------------------+  
    | Level   | Code | Message                                      |  
    +---------+------+----------------------------------------------+  
    | Warning | 1292 | Truncated incorrect DOUBLE value: '7788ab12' |  
    +---------+------+----------------------------------------------+  
    1 row in set (0.00 sec)  
      
    MySQL [test]> select * from emp where empno='ab7788';   ## 这个就相当于empno=0  
    Empty set (0.01 sec)  


3. 字符类型(varchar)

同样,针对测试表emp中的ename字段(varchar类型),上面有一辅助索引idx_ename,并且ename中有一个值是全数字的,若有这样的查询:



    select * from emp where ename=23456;  

上面的SQL会不会出现隐式转换呢?


下面实验证明:



    MySQL [test]> select * from emp where ename='23456';  
    +-------+-------+-----------+------+---------------------+------+------+--------+  
    | EMPNO | ENAME | JOB       | MGR  | HIREDATE            | SAL  | COMM | DEPTNO |  
    +-------+-------+-----------+------+---------------------+------+------+--------+  
    |  7936 | 23456 | BOSCO-DBA | 7788 | 2014-09-13 16:13:56 | 2450 |  800 |     10 |  
    +-------+-------+-----------+------+---------------------+------+------+--------+  
    1 row in set (0.00 sec)  
      
    MySQL [test]> explain select * from emp where ename='23456';   ## 正常来说,可以使用到索引idx_ename  
    +----+-------------+-------+------+---------------+-----------+---------+-------+------+-----------------------+  
    | id | select_type | table | type | possible_keys | key       | key_len | ref   | rows | Extra                 |  
    +----+-------------+-------+------+---------------+-----------+---------+-------+------+-----------------------+  
    |  1 | SIMPLE      | emp   | ref  | idx_ename     | idx_ename | 47      | const |    1 | Using index condition |  
    +----+-------------+-------+------+---------------+-----------+---------+-------+------+-----------------------+  
    1 row in set (0.00 sec)  




    MySQL [test]> select * from emp where ename=23456;   ## 当varchar类型不对时,仍然是可以查出结果  
    +-------+-------+-----------+------+---------------------+------+------+--------+  
    | EMPNO | ENAME | JOB       | MGR  | HIREDATE            | SAL  | COMM | DEPTNO |  
    +-------+-------+-----------+------+---------------------+------+------+--------+  
    |  7936 | 23456 | BOSCO-DBA | 7788 | 2014-09-13 16:13:56 | 2450 |  800 |     10 |  
    +-------+-------+-----------+------+---------------------+------+------+--------+  
    1 row in set, 14 warnings (0.00 sec)  
      
    MySQL [test]> explain select * from emp where ename=23456;   ## 当varchar类型不匹配时,索引无效了,选择了全表扫描  
    +----+-------------+-------+------+---------------+------+---------+------+------+-------------+  
    | id | select_type | table | type | possible_keys | key  | key_len | ref  | rows | Extra       |  
    +----+-------------+-------+------+---------------+------+---------+------+------+-------------+  
    |  1 | SIMPLE      | emp   | ALL  | idx_ename     | NULL | NULL    | NULL |   15 | Using where |  
    +----+-------------+-------+------+---------------+------+---------+------+------+-------------+  
    1 row in set (0.00 sec)  


可见,如果是字符类型,当出现类型不一致时,是会影响索引的使用的,会产生隐式转换的。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.iyunv.com/thread-24801-1-1.html 上篇帖子: MYSQL使用指南DOS下操作 下篇帖子: mysql远程连接:ERROR 1130 (HY000): Host '*.*.*.*' is not allowed to connect to this M
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表