加入收藏 | 设为首页 | 会员中心 | 我要投稿 辽源站长网 (https://www.0437zz.com/)- 云专线、云连接、智能数据、边缘计算、数据安全!
当前位置: 首页 > 站长学院 > MySql教程 > 正文

你不会还在用这8个错误的SQL写法吧?

发布时间:2019-09-22 04:23:24 所属栏目:MySql教程 来源:Java架构师追风
导读:1、LIMIT 语句 分页查询是最常用的场景之一,但也通常也是最容易出问题的地方。比如对于下面简单的语句,一般 DBA 想到的办法是在 type, name, create_time 字段上加组合索引。这样条件排序都能有效的利用到索引,性能迅速提升。 SELECT* FROMoperation WH

再检查执行计划:子查询物化后(select_type=DERIVED)参与 JOIN。虽然估算行扫描仍然为90万,但是利用了索引以及 LIMIT 子句后,实际执行时间变得很小。

  1. +----+-------------+------------+--------+---------------+---------+---------+-------+--------+----------------------------------------------------+ 
  2. | id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra | 
  3. +----+-------------+------------+--------+---------------+---------+---------+-------+--------+----------------------------------------------------+ 
  4. | 1 | PRIMARY | <derived2> | ALL | NULL | NULL | NULL | NULL | 15 | Using temporary; Using filesort | 
  5. | 1 | PRIMARY | u | eq_ref | PRIMARY | PRIMARY | 4 | o.uid | 1 | NULL | 
  6. | 1 | PRIMARY | p | ALL | PRIMARY | NULL | NULL | NULL | 6 | Using where; Using join buffer (Block Nested Loop) | 
  7. | 2 | DERIVED | o | index | NULL | idx_1 | 5 | NULL | 909112 | Using where | 
  8. +----+-------------+------------+--------+---------------+---------+---------+-------+--------+----------------------------------------------------+ 

8、中间结果集下推

再来看下面这个已经初步优化过的例子(左连接中的主表优先作用查询条件):

  1. SELECT a.*,  
  2.  c.allocated  
  3. FROM (  
  4.  SELECT resourceid  
  5.  FROM my_distribute d  
  6.  WHERE isdelete = 0  
  7.  AND cusmanagercode = '1234567'  
  8.  ORDER BY salecode limit 20) a  
  9. LEFT JOIN  
  10.  (  
  11.  SELECT resourcesid, sum(ifnull(allocation, 0) * 12345) allocated  
  12.  FROM my_resources  
  13.  GROUP BY resourcesid) c  
  14. ON a.resourceid = c.resourcesid 

那么该语句还存在其它问题吗?不难看出子查询 c 是全表聚合查询,在表数量特别大的情况下会导致整个语句的性能下降。

其实对于子查询 c,左连接最后结果集只关心能和主表 resourceid 能匹配的数据。因此我们可以重写语句如下,执行时间从原来的2秒下降到2毫秒。

  1. SELECT a.*,  
  2.  c.allocated  
  3. FROM (  
  4.  SELECT resourceid  
  5.  FROM my_distribute d  
  6.  WHERE isdelete = 0  
  7.  AND cusmanagercode = '1234567'  
  8.  ORDER BY salecode limit 20) a  
  9. LEFT JOIN  
  10.  (  
  11.  SELECT resourcesid, sum(ifnull(allocation, 0) * 12345) allocated  
  12.  FROM my_resources r,  
  13.  (  
  14.  SELECT resourceid  
  15.  FROM my_distribute d  
  16.  WHERE isdelete = 0  
  17.  AND cusmanagercode = '1234567'  
  18.  ORDER BY salecode limit 20) a  
  19.  WHERE r.resourcesid = a.resourcesid  
  20.  GROUP BY resourcesid) c  
  21. ON a.resourceid = c.resourcesid 

但是子查询 a 在我们的SQL语句中出现了多次。这种写法不仅存在额外的开销,还使得整个语句显的繁杂。使用 WITH 语句再次重写:

  1. WITH a AS  
  2. (  
  3.  SELECT resourceid  
  4.  FROM my_distribute d  
  5.  WHERE isdelete = 0  
  6.  AND cusmanagercode = '1234567'  
  7.  ORDER BY salecode limit 20) 
  8. SELECT a.*,  
  9.  c.allocated  
  10. FROM a  
  11. LEFT JOIN  
  12.  (  
  13.  SELECT resourcesid, sum(ifnull(allocation, 0) * 12345) allocated  
  14.  FROM my_resources r,  
  15.  a  
  16.  WHERE r.resourcesid = a.resourcesid  
  17.  GROUP BY resourcesid) c  
  18. ON a.resourceid = c.resourcesid 

总结

数据库编译器产生执行计划,决定着SQL的实际执行方式。但是编译器只是尽力服务,所有数据库的编译器都不是尽善尽美的。

上述提到的多数场景,在其它数据库中也存在性能问题。了解数据库编译器的特性,才能避规其短处,写出高性能的SQL语句。

程序员在设计数据模型以及编写SQL语句时,要把算法的思想或意识带进来。

编写复杂SQL语句要养成使用 WITH 语句的习惯。简洁且思路清晰的SQL语句也能减小数据库的负担 。

(编辑:辽源站长网)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

推荐文章
    热点阅读