依睛(IT blog) 我回来了,PHP<-->C/C++ LINUX

笨鸟

统计

积分与排名

友情连接

最新评论

MySQL如何优化 WHERE 子句?

MySQL如何优化 WHERE 子句?
作者:  来源:   发布日期:2008-06-13  

  这个章节讲述了优化程序如何处理 WHERE 子句。例子中使用了 SELECT 语句,但是在 DELETE 和 UPDATE 语句中对 WHERE 子句的优化是一样的。注意,关于MySQL优化的工作还在继续,因此本章节还没结束。MySQL做了很多优化工作,而不仅仅是文档中提到的这些。

  MySQL的一些优化做法如下:

  去除不必要的括号:

  ((a AND b) AND c OR (((a AND b) AND (c AND d))))

  -> (a AND b AND c) OR (a AND b AND c AND d)

  展开常量:

  (a

  -> b>5 AND b=c AND a=5

  去除常量条件(在展开常量时需要):

  (B>=5 AND B=5) OR (B=6 AND 5=5) OR (B=7 AND 5=6)

  -> B=5 OR B=6

  常量表达示在索引中只计算一次

  在单独一个表上做 COUNT(*) 而不使用 WHERE 时, 对于 MyISAM 和 HEAP 表就会直接从表信息中检索结果。在单独一个表上做任何表 NOT NULL 达式查询时也是这样做。

  预先探测无效的常量表达式。MySQL会快速探测一些不可能的 SELECT 语句并且不返回任何记录。

  当没用 GROUP BY 或分组函数时,HAVING 和 WHERE 合并(COUNT(), MIN() 等也是如此)。

  为表连接中的每个表构造一个简洁的 WHERE 语句,以得到更快的 WHERE 计算值并且尽快跳过记录。

  查询中所有的常量表都会比其他表更早读取。一个常量表符合以下几个条件:

  空表或者只有一条记录。

  与在一个 UNIQUE 索引、或一个 PRIMARY KEY 的 WHERE 子句一起使用的表,这里所有的索引部分和常数表达式做比较并且索引部分被定义为 NOT NULL。

  以下的几个表都会被当成常量表:


  SELECT * FROM t WHERE primary_key=1;
  SELECT * FROM t1,t2
  WHERE t1.primary_key=1 AND t2.primary_key=t1.id;

  MySQL会进各种可能找到表连接最好的连接方法。 如果在 ORDER BY 和 GROUP BY 子句中的所有字段都来自同一个表的话,那么在连接时这个表就会优先处理。

  如果有 ORDER BY 子句和一个不同的 GROUP BY 子句,或者如果 ORDER BY 或 GROUP BY 中的字段都来自其他的表而非连接顺序中的第一个表的话,就会创建一个临时表了。

  如果使用 SQL_SMALL_RESULT,MySQL就会使用内存临时表了。

  所有的表索引都会查询,最好的情况就是所有的索引都会被用到,除非优化程序认为全表扫描的效率更高。同时,数据表扫描是基于判断最好的索引范围超过数据表的30%。 现在,优化程序复杂多了,它基于对一些附加因素的估计,例如表大小,记录总数,I/O块大小,因此就不能根据一个固定的百分比来决定是选择使用索引还是直接扫描数据表。

  在某些情况下,MySQL可以直接从索引中取得记录而无需查询数据文件。如果所有在索引中使用的字段都是数字类型的话,只需要用索引树就能完成查询。

  每条记录输出之前,那些没有匹配 HAVING 子句的就会被跳过。

  以下几个查询速度非常快:


  SELECT COUNT(*) FROM tbl_name;
  SELECT MIN(key_part1),MAX(key_part1) FROM tbl_name;
  SELECT MAX(key_part2) FROM tbl_name
  WHERE key_part1=constant;
  SELECT ... FROM tbl_name
  ORDER BY key_part1,key_part2,... LIMIT 10;
  SELECT ... FROM tbl_name
  ORDER BY key_part1 DESC, key_part2 DESC, ... LIMIT 10;

  以下几个查询都是使用索引树,假使那些索引字段都是数字型:


  SELECT key_part1,key_part2 FROM tbl_name WHERE key_part1=val;
  SELECT COUNT(*) FROM tbl_name
  WHERE key_part1=val1 AND key_part2=val2;
  SELECT key_part2 FROM tbl_name GROUP BY key_part1;

  以下几个查询使用索引来取得经过顺序排序后的记录而无需经过独立的排序步骤: SELECT ... FROM tbl_name
  ORDER BY key_part1,key_part2,... ;
  SELECT ... FROM tbl_name
  ORDER BY key_part1 DESC, key_part2 DESC, ... ;

posted on 2009-01-03 15:33 向左向右走 阅读(96) 评论(0)  编辑 收藏 引用

只有注册用户登录后才能发表评论。