MYSQL WHERE語句優(yōu)化
更新時(shí)間:2009年03月13日 22:45:04 作者:
where優(yōu)化主要是在SELECT中,因?yàn)樗麄冏钪饕窃谀抢锸褂茫峭瑯拥膬?yōu)化也可被用于DELETE和UPDATE語句。
但請(qǐng)注意,下面的優(yōu)化并不是完全的。MYSQL實(shí)施了許多優(yōu)化,但我沒時(shí)間全部測(cè)試.
MySQL的一些優(yōu)化列在下面:
刪除不必要的括號(hào):
((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)
常數(shù)調(diào)入:
(a<b AND b=c) AND a=5
-> b>5 AND b=c AND a=5
刪除常數(shù)條件:
(B>=5 AND B=5) OR (B=6 AND 5=5) OR (B=7 AND 5=6)
-> B=5 OR B=6
索引使用的常數(shù)表達(dá)式僅計(jì)算一次。
在一個(gè)單個(gè)表上的沒有一個(gè)WHERE的COUNT(*)直接從表中檢索信息。當(dāng)僅使用一個(gè)表時(shí),對(duì)任何NOT NULL表達(dá)式也這樣做。
無效常數(shù)表達(dá)式的早期檢測(cè)。MySQL快速檢測(cè)某些SELECT語句是不可能的并且不返回行。
如果你不使用GROUP BY或分組函數(shù)(COUNT()、MIN()……),HAVING與WHERE合并。
為每個(gè)子聯(lián)結(jié)(sub join),構(gòu)造一個(gè)更簡(jiǎn)單的WHERE以得到一個(gè)更快的WHERE計(jì)算并且也盡快跳過記錄。
所有常數(shù)的表在查詢中的在其他任何表之前被讀出。
一個(gè)常數(shù)的表是:
一個(gè)空表或一個(gè)有1行的表。
與在一個(gè)UNIQUE索引、或一個(gè)PRIMARY KEY的WHERE子句一起使用的表,這里所有的索引部分使用一個(gè)常數(shù)表達(dá)式并且索引部分被定義為NOT NULL。
所有下列的表用作常數(shù)表:
mysql> SELECT * FROM t WHERE primary_key=1;
mysql> SELECT * FROM t1,t2 WHERE t1.primary_key=1 AND t2.primary_key=t1.id;
對(duì)聯(lián)結(jié)表的最好聯(lián)結(jié)組合是通過嘗試所有可能性來找到:(。如果所有在ORDER BY和GROUP BY的列來自同一個(gè)表,那么當(dāng)聯(lián)結(jié)時(shí),該表首先被選中。
如果你使用SQL_SMALL_RESULT,MySQL將使用一個(gè)在內(nèi)存中的表。
如果有一個(gè)ORDER BY子句和一個(gè)不同的GROUP BY子句,或如果ORDER BY或GROUP BY包含不是來自聯(lián)結(jié)隊(duì)列中的第一個(gè)表的其他表的列,創(chuàng)建一個(gè)臨時(shí)表。
因?yàn)镈ISTINCT被變換到在所有的列上的一個(gè)GROUP BY,DISTINCT與ORDER BY結(jié)合也將在許多情況下需要一張臨時(shí)表。
每個(gè)表的索引被查詢并且使用跨越少于30% 的行的索引。如果這樣的索引沒能找到,將使用一個(gè)快速的表掃描。
在一些情況下,MySQL能從索引中讀出行,甚至不用查詢數(shù)據(jù)文件。如果索引使用的所有列是數(shù)字的,那么只有索引樹被用來解答查詢。
在每個(gè)記錄被輸出前,那些不匹配HAVING子句的行將被跳過。
下面是一些快速的查詢例子:
mysql> SELECT COUNT(*) FROM tbl_name;
mysql> SELECT MIN(key_part1),MAX(key_part1) FROM tbl_name;
mysql> SELECT MAX(key_part2) FROM tbl_name
WHERE key_part_1=constant;
mysql> SELECT ... FROM tbl_name
ORDER BY key_part1,key_part2,... LIMIT 10;
mysql> SELECT ... FROM tbl_name
ORDER BY key_part1 DESC,key_part2 DESC,... LIMIT 10;
下列查詢僅使用索引樹就可解決(假設(shè)索引列是數(shù)字的):
mysql> SELECT key_part1,key_part2 FROM tbl_name WHERE key_part1=val;
mysql> SELECT COUNT(*) FROM tbl_name
WHERE key_part1=val1 AND key_part2=val2;
mysql> SELECT key_part2 FROM tbl_name GROUP BY key_part1;
下列查詢使用索引以排序順序檢索,不用一次另外的排序:
mysql> SELECT ... FROM tbl_name ORDER BY key_part1,key_part2,...
mysql> SELECT ... FROM tbl_name ORDER BY key_part1 DESC,key_part2 DESC,...
MySQL的一些優(yōu)化列在下面:
刪除不必要的括號(hào):
((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)
常數(shù)調(diào)入:
(a<b AND b=c) AND a=5
-> b>5 AND b=c AND a=5
刪除常數(shù)條件:
(B>=5 AND B=5) OR (B=6 AND 5=5) OR (B=7 AND 5=6)
-> B=5 OR B=6
索引使用的常數(shù)表達(dá)式僅計(jì)算一次。
在一個(gè)單個(gè)表上的沒有一個(gè)WHERE的COUNT(*)直接從表中檢索信息。當(dāng)僅使用一個(gè)表時(shí),對(duì)任何NOT NULL表達(dá)式也這樣做。
無效常數(shù)表達(dá)式的早期檢測(cè)。MySQL快速檢測(cè)某些SELECT語句是不可能的并且不返回行。
如果你不使用GROUP BY或分組函數(shù)(COUNT()、MIN()……),HAVING與WHERE合并。
為每個(gè)子聯(lián)結(jié)(sub join),構(gòu)造一個(gè)更簡(jiǎn)單的WHERE以得到一個(gè)更快的WHERE計(jì)算并且也盡快跳過記錄。
所有常數(shù)的表在查詢中的在其他任何表之前被讀出。
一個(gè)常數(shù)的表是:
一個(gè)空表或一個(gè)有1行的表。
與在一個(gè)UNIQUE索引、或一個(gè)PRIMARY KEY的WHERE子句一起使用的表,這里所有的索引部分使用一個(gè)常數(shù)表達(dá)式并且索引部分被定義為NOT NULL。
所有下列的表用作常數(shù)表:
mysql> SELECT * FROM t WHERE primary_key=1;
mysql> SELECT * FROM t1,t2 WHERE t1.primary_key=1 AND t2.primary_key=t1.id;
對(duì)聯(lián)結(jié)表的最好聯(lián)結(jié)組合是通過嘗試所有可能性來找到:(。如果所有在ORDER BY和GROUP BY的列來自同一個(gè)表,那么當(dāng)聯(lián)結(jié)時(shí),該表首先被選中。
如果你使用SQL_SMALL_RESULT,MySQL將使用一個(gè)在內(nèi)存中的表。
如果有一個(gè)ORDER BY子句和一個(gè)不同的GROUP BY子句,或如果ORDER BY或GROUP BY包含不是來自聯(lián)結(jié)隊(duì)列中的第一個(gè)表的其他表的列,創(chuàng)建一個(gè)臨時(shí)表。
因?yàn)镈ISTINCT被變換到在所有的列上的一個(gè)GROUP BY,DISTINCT與ORDER BY結(jié)合也將在許多情況下需要一張臨時(shí)表。
每個(gè)表的索引被查詢并且使用跨越少于30% 的行的索引。如果這樣的索引沒能找到,將使用一個(gè)快速的表掃描。
在一些情況下,MySQL能從索引中讀出行,甚至不用查詢數(shù)據(jù)文件。如果索引使用的所有列是數(shù)字的,那么只有索引樹被用來解答查詢。
在每個(gè)記錄被輸出前,那些不匹配HAVING子句的行將被跳過。
下面是一些快速的查詢例子:
mysql> SELECT COUNT(*) FROM tbl_name;
mysql> SELECT MIN(key_part1),MAX(key_part1) FROM tbl_name;
mysql> SELECT MAX(key_part2) FROM tbl_name
WHERE key_part_1=constant;
mysql> SELECT ... FROM tbl_name
ORDER BY key_part1,key_part2,... LIMIT 10;
mysql> SELECT ... FROM tbl_name
ORDER BY key_part1 DESC,key_part2 DESC,... LIMIT 10;
下列查詢僅使用索引樹就可解決(假設(shè)索引列是數(shù)字的):
mysql> SELECT key_part1,key_part2 FROM tbl_name WHERE key_part1=val;
mysql> SELECT COUNT(*) FROM tbl_name
WHERE key_part1=val1 AND key_part2=val2;
mysql> SELECT key_part2 FROM tbl_name GROUP BY key_part1;
下列查詢使用索引以排序順序檢索,不用一次另外的排序:
mysql> SELECT ... FROM tbl_name ORDER BY key_part1,key_part2,...
mysql> SELECT ... FROM tbl_name ORDER BY key_part1 DESC,key_part2 DESC,...
相關(guān)文章
mysql實(shí)現(xiàn)查詢結(jié)果導(dǎo)出csv文件及導(dǎo)入csv文件到數(shù)據(jù)庫(kù)操作
這篇文章主要介紹了mysql實(shí)現(xiàn)查詢結(jié)果導(dǎo)出csv文件及導(dǎo)入csv文件到數(shù)據(jù)庫(kù)操作,結(jié)合實(shí)例形式分析了mysql相關(guān)數(shù)據(jù)庫(kù)導(dǎo)出、導(dǎo)入語句使用方法及操作注意事項(xiàng),需要的朋友可以參考下2018-07-07
MySQL數(shù)據(jù)庫(kù)遭到攻擊篡改(使用備份和binlog進(jìn)行數(shù)據(jù)恢復(fù))
這篇文章主要介紹了MySQL數(shù)據(jù)庫(kù)遭到攻擊篡改(使用備份和binlog進(jìn)行數(shù)據(jù)恢復(fù)),需要的朋友可以參考下2016-04-04
MySQL實(shí)現(xiàn)字段分割一行轉(zhuǎn)多行的示例代碼
這篇文章主要介紹了MySQL實(shí)現(xiàn)字段分割一行轉(zhuǎn)多行的示例代碼,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-07-07
mysql 控制臺(tái)程序的提示符 prompt 字符串設(shè)置
mysql 控制臺(tái)程序的提示符 prompt 字符串設(shè)置,學(xué)習(xí)mysql的朋友可以參考下。2011-08-08
Mysql ID生成策略的三種方法選擇及優(yōu)缺點(diǎn)
mysql ID生成策略一般常用的有三種,包括自增、UUID 以及雪花算法,本文主要介紹了Mysql ID生成策略的三種方法選擇及優(yōu)缺點(diǎn),具有一定的參考價(jià)值,感興趣的可以了解一下2024-06-06
mysql如何實(shí)現(xiàn)多行查詢結(jié)果合并成一行
利用函數(shù):group_concat(),實(shí)現(xiàn)一個(gè)ID對(duì)應(yīng)多個(gè)名稱時(shí),原本為多行數(shù)據(jù),把名稱合并成一行2013-12-12
MySQL實(shí)戰(zhàn)文章(非常全的基礎(chǔ)入門類教程)
半個(gè)月時(shí)間把MySQL重新鞏固了一遍,梳理了一篇幾萬字超硬核文章,想學(xué)習(xí)mysql的朋友可以看看2023-05-05

