当前位置:  数据库>oracle

MySQL和Oracle中的隐式转换

    来源: 互联网  发布时间:2017-06-19

    本文导语: 今天在处理一个问题的时候,需要根据其他部门提供的sql语句对一个表中的数据进行了筛查。 语句类似下面的形式> SELECT MAX_LEVEL,LOGOUT_TIME,CURRENT_DATE AS NOWTIME,cn_master FROM t_test_october_back_a WHERE ID in ( 100, 200, 300, 400, 500) ; +-----------+-...

今天在处理一个问题的时候,需要根据其他部门提供的sql语句对一个表中的数据进行了筛查。
 语句类似下面的形式
> SELECT MAX_LEVEL,LOGOUT_TIME,CURRENT_DATE AS NOWTIME,cn_master FROM t_test_october_back_a WHERE ID in ( 100, 200, 300, 400, 500) ;
 +-----------+---------------+------------+-----------+
 | MAX_LEVEL | LOGOUT_TIME  | NOWTIME    | ID|
 +-----------+---------------+------------+-----------+
 |        1 | 1440407918000 | 2015-08-31 | 100|
 |        100| 1441009281000 | 2015-08-31 | 200|
 |        1 | 1440408002000 | 2015-08-31 | 300|
 +-----------+---------------+------------+-----------+
 x rows in set, 65535 warnings (10.98 sec)
本来这一个简单查询就完成了,也得到了业务部门需要的数据情况,但是查看最后一行的内容,还是有些蹊跷。如果观察仔细,对于这种id的数据查询,走索引的话,绝对不会再10秒左右,这是第一个奇怪的地方,第二个奇怪的地方就是65535 warnings,一个简单查询怎么会有这么多的warnings
最开始怕show warnings的时候会一下子显示出来6万多行数据,还小小担心了一下,结果输出的结果只有64行。
>show warnings;
 +---------+------+----------------------------------------------------------------+
 | Level  | Code | Message                                                        |
 +---------+------+----------------------------------------------------------------+
 | Warning | 1292 | Truncated incorrect DOUBLE value: '3506996@abc.com'      |
 | Warning | 1292 | Truncated incorrect DOUBLE value: '28366@abc.com'    |
 | Warning | 1292 | Truncated incorrect DOUBLE value: '81700700@abc.com'    |
 | Warning | 1292 | Truncated incorrect DOUBLE value: '391112900@abc.com'    |
 | Warning | 1292 | Truncated incorrect DOUBLE value: '867964771@abc.com'    |
 ...
 64 rows in set (0.00 sec)
查看建表语句,发现这个id列是varchar类型的。
>show create table  t_test_october_back_a;
 | t_tl_october_back_a | CREATE TABLE `t_tl_october_back_a` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `id` varchar(60) NOT NULL,
  `area_server` varchar(80) NOT NULL,
  `max_level` tinyint(4) NOT NULL,
  `logout_time` bigint(20) NOT NULL,
  PRIMARY KEY (`id`),
  UNIQUE KEY `idx_test_october_back_cn_master` (`id`)
 ) ENGINE=InnoDB AUTO_INCREMENT=2042252493 DEFAULT CHARSET=gbk |

我们就顺着错误信息来看看,把警告中的部分直接作为参数,发现查询的时间极快。
>select *from t_tl_october_back_a where cn_master ='867964771@abc.com';
 +-------+-------------------------+---------------------------+-----------+---------------+
 | id    | cn_master              | area_server              | max_level | logout_time  |
 +-------+-------------------------+---------------------------+-----------+---------------+
 | 18723 | 867964771@abc.com | abcd-abcd            |      104 | 1434446979000 |
 +-------+-------------------------+---------------------------+-----------+---------------+
 1 row in set (0.00 sec)
对此我们来通过执行计划来简单对比一下。
 如果使用数字的方式,会走全表扫描
> explain SELECT MAX_LEVEL,LOGOUT_TIME FROM t_tl_october_back_a WHERE CN_MASTER in ( 100, 200);
 +----+-------------+---------------------+------+-------------------------------+------+---------+------+----------+-------------+
 | id | select_type | table              | type | possible_keys                | key  | key_len | ref  | rows    | Extra      |
 +----+-------------+---------------------+------+-------------------------------+------+---------+------+----------+-------------+
 |  1 | SIMPLE      | t_test_october_back_a | ALL  | idx_test_october_back_cn_master | NULL | NULL    | NULL | 28597841 | Using where |
 +----+-------------+---------------------+------+-------------------------------+------+---------+------+----------+-------------+
 1 row in set (0.00 sec)
而如果使用字符的形式,执行计划就会走索引,执行效果就是预期的样子。
>explain SELECT MAX_LEVEL,LOGOUT_TIME FROM t_tl_october_back_a WHERE CN_MASTER = '100' ;
 +----+-------------+---------------------+-------+-------------------------------+-------------------------------+---------+-------+------+-------+
 | id | select_type | table              | type  | possible_keys                | key                          | key_len | ref  | rows | Extra |
 +----+-------------+---------------------+-------+-------------------------------+-------------------------------+---------+-------+------+-------+
 |  1 | SIMPLE      | t_test_october_back_a | const | idx_test_october_back_cn_master | idx_test_october_back_cn_master | 122    | const |    1 |      |
 +----+-------------+---------------------+-------+-------------------------------+-------------------------------+---------+-------+------+-------+
 1 row in set (0.00 sec)
通过上面的例子可以看到,在查询的时候抛出的警告,其实就是在做类型转换的时候本来输出的是数字类型,就会尝试做隐式转换,而那个65535只是一个最大限制而已,表中的数据其实已经远远超过千万。
 这个时候我们大体感受到了隐式转换在MySQL中的一些影响,我们来看看在MySQL 5.6和Oracle中的表现如何。
###############
MYSQL 5.6
 mysql> select version();
 +-------------------------------------------+
 | version()                                |
 +-------------------------------------------+
 | 5.6.23-enterprise-commercial-advanced-log |
 +-------------------------------------------+
 mysql> create table test (id1 int,id2 varchar(10));
 mysql> insert into test values(1,'1');
 mysql> insert into test values(2,'2');
 mysql> insert into test values(3,'3');
 mysql> commit;
 mysql> create index idx_id1 on test(id1);
 mysql> create index idx_id2 on test(id2);
隐式转换,有数字转换为字符的时候,直接走了索引扫描
mysql> explain select * from test where id1='1';
 +----+-------------+-------+------+---------------+---------+---------+-------+------+-------+
 | id | select_type | table | type | possible_keys | key    | key_len | ref  | rows | Extra |
 +----+-------------+-------+------+---------------+---------+---------+-------+------+-------+
 |  1 | SIMPLE      | test  | ref  | idx_id1      | idx_id1 | 5      | const |    1 | NULL  |
 +----+-------------+-------+------+---------------+---------+---------+-------+------+-------+
 1 row in set (0.01 sec)

隐式转换,由字符转换为数字的时候,直接走了全表扫描
mysql> explain select *from test where id2=2;
 +----+-------------+-------+------+---------------+------+---------+------+------+-------------+
 | id | select_type | table | type | possible_keys | key  | key_len | ref  | rows | Extra      |
 +----+-------------+-------+------+---------------+------+---------+------+------+-------------+
 |  1 | SIMPLE      | test  | ALL  | idx_id2      | NULL | NULL    | NULL |    3 | Using where |
 +----+-------------+-------+------+---------------+------+---------+------+------+-------------+
 1 row in set (0.00 sec)


###############
 Oracle 11gR2

 sqlplus -v
 SQL*Plus: Release 10.2.0.3.0 - Production

SQL> create table test (id1 number,id2 varchar2(10));
 SQL> insert into test values(1,'1');
 SQL> begin                 
  2  for i in 1..100 loop
  3  insert into test values(i,chr(39)||i||chr(39));
  4  end loop;
  5  commit;
  6  end;
  7  /
 PL/SQL procedure successfully completed.
 SQL> exec dbms_stats.gather_table_stats('TEST','TEST',CASCADE=>TRUE);
隐式转换,由数字转换为字符的时候,直接走了索引扫描
SQL> explain plan for select *from test where id1='2';
 SQL> select *from table(dbms_xplan.display);
 PLAN_TABLE_OUTPUT
 --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 Plan hash value: 3847161316
 ---------------------------------------------------------------------------------------
 | Id  | Operation                  | Name    | Rows  | Bytes | Cost (%CPU)| Time    |
 ---------------------------------------------------------------------------------------
 |  0 | SELECT STATEMENT            |        |    1 |    7 |    2  (0)| 00:00:01 |
 |  1 |  TABLE ACCESS BY INDEX ROWID| TEST    |    1 |    7 |    2  (0)| 00:00:01 |
 |*  2 |  INDEX RANGE SCAN          | IDX_ID1 |    1 |      |    1  (0)| 00:00:01 |
 ---------------------------------------------------------------------------------------
 Predicate Information (identified by operation id):
 PLAN_TABLE_OUTPUT
    2 - access("ID1"=2)

隐式转换,由字符转换为数字的时候,直接走了全表扫��
SQL> explain plan for select *from test where id2=3;
 SQL> select *from table(dbms_xplan.display);
 PLAN_TABLE_OUTPUT
 --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 Plan hash value: 1357081020
 --------------------------------------------------------------------------
 | Id  | Operation        | Name | Rows  | Bytes | Cost (%CPU)| Time    |
 --------------------------------------------------------------------------
 |  0 | SELECT STATEMENT  |      |    1 |    7 |    3  (0)| 00:00:01 |
 |*  1 |  TABLE ACCESS FULL| TEST |    1 |    7 |    3  (0)| 00:00:01 |
 --------------------------------------------------------------------------
 Predicate Information (identified by operation id):
 ---------------------------------------------------
 PLAN_TABLE_OUTPUT
 --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
    1 - filter(TO_NUMBER("ID2")=3)

可见在这个方面MySQL和Oracle中的表现是一致的,对于这种隐式转换还是要多加注意。


    
 
 

您可能感兴趣的文章:

  • mysql中字符串和时间互相转换的方法(自动转换及DATE_FORMAT函数)
  • Mysql 数字类型转换函数
  • linux下安装mysql字符集转换的问题
  • mysql中将null值转换为0的语句
  • 深入MYSQL字符数字转换的详解
  • mysql编码转换
  • 使用mss2sql工具将SqlServer转换为Mysql全记录
  • python3图片转换二进制存入mysql
  • 完美转换MySQL的字符集 解决查看utf8源文件中的乱码问题
  • mssql转换mysql的方法小结
  • MySQL转换Oracle的需要注意的七个事项
  • MySQL数据库向Oracle转换时注意若干问题
  • mysql 时间转换函数的使用方法第1/2页
  • php mysql中utf8编码汉字转换成拼音
  • 将mysql转换到oracle必须了解的50件事
  • Mysql中文汉字转拼音的实现(每个汉字转换全拼)
  • mysql 日期和时间格式转换实现语句
  • MySQL从MyISAM转换成InnoDB错误与常用解决办法
  • 将MySQL从MyISAM转换成InnoDB错误和解决办法
  • Oracle欲收购开源数据库MySQL未果
  • MySQL类似Oracle的dual虚拟表
  • 在linux下是MYSQL好还是ORACLE厉害?
  • Oracle与Mysql主键、索引及分页的区别小结
  • MySQL实现类似Oracle中的decode()函数的功能
  • Linux(redhat)下除了oracle / MySql 外还有哪些稍大型一点的 数据库
  • mysql仿oracle的decode效果查询
  • MySQL 移植到Oracle10 的问题
  • Oracle承诺一出 MySQL 5.5 版就发布
  • 解析mysql与Oracle update的区别
  • oracle与mysql的视频教程下载地址分享
  • 调查服务器系统BSD,SCO,Linux,windows?和数据库mysql,oracle,sqlserver?
  •  
    本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
    本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • mysql中如何查看最大连接数(max_connections)和修改最大连接数
  • 在 linux下输入"mysql"命令,进入mysql命令行,但出现“Can't connetc to local MySQL server thuough socket /var/lib/mysql/mysql.sock
  • Mysql查询错误:ERROR:no query specified原因
  • MySQL 重装MySQL后, mysql服务无法启动
  • php安装完成后如何添加mysql扩展
  • 为什么用linux安装盘安装了mysql后,启动mysql,提示找不到mysql.sock文件?
  • mysql中查询当前正在运行的SQL语句并找出mysql中运行慢的sql语句
  • 請教,在redhat linux7.2+mysql 中,系統提示mysql已啟動,網頁卻不能訪問mysql?
  • Myeclipse中自带Tomcat的JDBC连接池配置(mysql和mssql)
  • 求解释: useradd -g mysql mysql -d /home/mysql -s /sbin/nologin
  • MySQL Workbench的下载安装与使用教程
  • 在Linux内安装了Mysql,无法进入Mysql.
  • php中内置的mysql数据库连接驱动mysqlnd简介及mysqlnd的配置安装方式
  • 怎样在linux终端输入mysql直接进入mysql?
  • VS2012+MySQL+SilverLight5的MVVM开发模式介绍
  • c++中关于#include <mysql/mysql.h>的问题?
  • MySQL索引基本知识
  • mysql -u root mysql 怎么解释
  • Mysql设置查询条件(where)查询字段为NULL
  • mm.mysql那里可以下载?www.mysql.com根本下载不了。谢谢了
  • java将类序列化并存储到mysql(使用hibernate)
  • MySQL集群 MySQL Cluster


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3