MySQL14-为什么SQL语句逻辑相同,性能却差异巨大?

案例一:条件字段函数操作

不要对字段进行函数计算操作。比如下面的就不行

select * from tradelog where id + 1 = 10000;
select count(*) from tradelog where month(t_modified)=7;

为什么不能对条件字段做计算?–>无法用到B+树的快速定位能力
因为如果对字段做了函数计算, 就用不上索引了, 这是MySQL的规定。而我们上面的id和t_modified都有索引,但是显然我们都对它们进行了操作,这样就无法用到B+树的快速定位能力,也就是说对对索引字段做函数操作, 可能会破坏索引值的有序性, 因此优化器就决定放弃走树搜索功能。所以等SQL语句执行了很久才会返回结果

select count(*) from tradelog where month(t_modified)=7;

在这个语句中,放弃了树搜索功能, 优化器可以选择遍历主键索引, 也可以选择遍历索引t_modified, 优化器对比索引大小后发现, 索引t_modified更小, 遍历这个索引比遍历主键索引来得更快。 因此最终还是会选择索引t_modified。

我们使用explain命令, 查看一下这条SQL语句的执行结果。
在这里插入图片描述
key="t_modified"表示的是, 使用了t_modified这个索引; 我在测试表数据中插入了10万行数据, rows=100335, 说明这条语句扫描了整个索引的所有值; Extra字段的Using index, 表示的是使用了覆盖索引。
也就是说, 由于在t_modified字段加了month()函数操作, 导致了全索引扫描导致执行时间过长

如何运用B+树的快速定位能力?
为了能够用上索引的快速定位能力, 我们就要把SQL语句改成基于字段本身的范围查询

mysql> select count(*) from tradelog where
-> (t_modified >= '2016-7-1' and t_modified<'2016-8-1') or
-> (t_modified >= '2017-7-1' and t_modified<'2017-8-1') or
-> (t_modified >= '2018-7-1' and t_modified<'2018-8-1');

案例二:隐式类型转换

数据类型与字段类型不同的,将导致全表扫描。
数据类型转换的规则是什么?
数据库中数据类型很多,涉及到的类型转换也很多,比如字符串和数据之间的比较,需要将字符串转换成数字才能比较,比如:

mysql> select * from tradelog where tradeid=110717;

tradeid的字段类型是varchar(32), 而输入的参数却是整型, 所以需要做类型转换。
交易编号tradeid这个字段上, 本来就有索引, 但是explain的结果却显示, 这条语句需要走全表扫描。

为什么有数据类型转换, 就需要走全索引扫描?
我们知道了数据类型转换,在MySQL中, 字符串和数字做比较的话, 是将字符串转换成数字。

mysql> select * from tradelog where tradeid=110717;

那么上面这条语句对优化器而言相当于

mysql> select * from tradelog where CAST(tradid AS signed int) = 110717;

这条语句触发了我们上面说到的规则: 对索引字段做函数操作, 优化器会放弃走树搜索功能,所以只能走全表查询。

案例三:隐式字符编码转换

两张表编码格式不一致也会导致全表查询。
什么是字符编码?
所谓的字符编码就是两个表选择的编码字符集不同,比如一个表选择utf-8,另一个表选择utf8mb4,当表连接查询的时候用不上关联字段的索引,先把utf8字符串转成utf8mb4字符集, 再做比较。 所以会导致执行的速度很慢。

比如这条语句

mysql> select d.* from tradelog l, trade_detail d where d.tradeid=l.tradeid and l.id=2;

我们一起来看下这个结果:
在这里插入图片描述

  1. 第一行显示优化器会先在交易记录表tradelog上查到id=2的行, 这个步骤用上了主键索引rows=1表示只扫描一行;
  2. 第二行key=NULL, 表示没有用上交易详情表trade_detail上的tradeid索引, 进行了全表扫描。

了解它的执行过程

  1. 是根据id在tradelog表里找到L2这一行;
  2. 是从L2中取出tradeid字段的值;
  3. 是根据tradeid值到trade_detail表中查找条件匹配的行。 explain的结果里面第二行的key=NULL表示的就是, 这个过程是通过遍历主键索引(全表扫描) 的方式, 一个一个地判断tradeid的值是否匹配

你会发现第3步不符合我们的预期。 因为表trade_detail里tradeid字段上是有索引的, 我们本来是希望通过使用tradeid索引能够快速定位到等值的行。 但, 这里并没有。

有什么解决方法呢?
连表查询的时候两个连接字段使用join操作。便可使用tradeid索引。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 要将Oracle SQL语句转化为MySQL SQL语句,需要注意以下几点: 1. 日期和时间函数:MySQL和Oracle的日期和时间函数有些差异,需要对部分函数进行调整。例如,将Oracle的sysdate函数替换为MySQL的current_date函数,将Oracle的to_char函数替换为MySQL的date_format函数。 2. 字符串函数:Oracle和MySQL的字符串函数用法有些不同。例如,将Oracle的substr函数替换为MySQL的substring函数,将Oracle的instr函数替换为MySQL的locate函数。 3. 数据类型:某些数据类型在Oracle和MySQL中的表示方式不同,需要将其进行转换。例如,将Oracle的number类型转换为MySQL的decimal类型,将Oracle的varchar2类型转换为MySQL的varchar类型。 4. 分页查询:Oracle和MySQL分页查询的语法也有所差异。在Oracle中,可以使用rownum来进行分页,而在MySQL中,可以使用limit和offset来实现。因此,在转换查询语句时,需要将Oracle的rownum替换为MySQL的limit和offset。 5. NULL值处理:Oracle和MySQL对NULL值的处理方式也有些差异。在Oracle中,可以使用is null和is not null来判断NULL值,而在MySQL中,可以使用is null和is not null,也可以使用isnull函数。 总之,将Oracle SQL语句转化为MySQL SQL语句需要注意这些差异,并根据具体情况进行适当调整和转换。实际操作中,可以参考MySQL的官方文档和手册,以便获得更详细和准确的转换规则和示例。 ### 回答2: 将Oracle SQL语句转化为MySQL SQL语句通常需要注意以下几个方面: 1. 数据类型:Oracle和MySQL的数据类型有所不同,需要根据情况进行转换。例如,将Oracle的NUMBER数据类型转换为MySQL的INT或FLOAT类型。 2. 函数和操作符:Oracle和MySQL支持的函数和操作符有所不同,需要根据需求进行调整和替换。例如,将Oracle的TO_CHAR函数转换为MySQL的DATE_FORMAT函数。 3. 子查询和连接:Oracle和MySQL的子查询和连接语法也有所差异,需要进行适当调整。例如,将Oracle的子查询使用WITH语句转换为MySQL的内部子查询。 4. 约束和索引:Oracle和MySQL的约束和索引语法也有差异,需要进行相应修改。例如,将Oracle的UNIQUE约束转换为MySQL的UNIQUE关键字。 下面是一个具体的例子,将Oracle的SQL语句转换为MySQLSQL语句: Oracle SQL语句: SELECT emp_name, hire_date FROM employees WHERE emp_id = 100; MySQL SQL语句: SELECT emp_name, hire_date FROM employees WHERE emp_id = 100; 这是一个简单的例子,在实际转换过程中可能会遇到更多的复杂情况,需要根据具体需求进行相应的调整和修改。同时需要注意两种数据库差异,以免出现语法错误或逻辑错误。另外,还可以通过使用一些第三方工具来自动完成SQL语句的转换,可以节省时间和精力。 ### 回答3: 要将Oracle SQL语句转化为MySQL SQL语句,需要注意一些Oracle特有的函数、关键字和语法在MySQL中可能没有或有不同的实现。下面是一些常见的转化技巧: 1. 字符串连接: Oracle中使用"||"作为字符串连接符,而MySQL使用"CONCAT"函数。例如: Oracle:SELECT firstname || ' ' || lastname FROM employees; MySQL:SELECT CONCAT(firstname, ' ', lastname) FROM employees; 2. 日期格式化: Oracle中使用"TO_CHAR"函数进行日期格式化,而MySQL使用"DATE_FORMAT"函数。例如: Oracle:SELECT TO_CHAR(hiredate, 'YYYY/MM/DD') FROM employees; MySQL:SELECT DATE_FORMAT(hiredate, '%Y/%m/%d') FROM employees; 3. 行号: 在Oracle中,可以使用"ROWNUM"来获取行号,而MySQL需要使用子查询和变量来实现类似的功能。例如: Oracle:SELECT * FROM employees WHERE ROWNUM <= 10; MySQL:SELECT * FROM employees WHERE id <= (SELECT id FROM employees ORDER BY id LIMIT 1 OFFSET 9); 4. 分页查询: 在Oracle中使用"ROWNUM"和"FETCH FIRST"实现分页查询,而MySQL使用"LIMIT"和"OFFSET"。例如: Oracle:SELECT * FROM employees WHERE ROWNUM > 10 AND ROWNUM <= 20; MySQL:SELECT * FROM employees LIMIT 10 OFFSET 10; 5. 自增主键列: 在Oracle中使用"SEQUENCE"和"TRIGGER"来生成自增主键值,而MySQL使用"AUTO_INCREMENT"关键字。例如: Oracle:CREATE SEQUENCE seq_emp_id START WITH 1 INCREMENT BY 1; CREATE TRIGGER tr_emp_id BEFORE INSERT ON employees FOR EACH ROW BEGIN SELECT seq_emp_id.nextval INTO :NEW.id FROM dual; END; MySQL:CREATE TABLE employees (id INT PRIMARY KEY AUTO_INCREMENT, firstname VARCHAR(50), lastname VARCHAR(50)); 需要注意的是,这只是一些常见的转化技巧,具体的转化还需要根据具体的SQL语句数据库结构进行调整和适配。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值