MySQL怎樣最佳化WHERE子句(轉)
MySQL怎樣最佳化WHERE子句(轉)[@more@]where最佳化主要是在SELECT中,因為他們最主要是在那裡使用,但是同樣的最佳化也可被用於DELETE和UPDATE語句。
但請注意,下面的最佳化並不是完全的。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
索引使用的常數表示式僅計算一次。
在一個單個表上的沒有一個WHERE的COUNT(*)直接從表中檢索資訊。當僅使用一個表時,對任何NOT NULL表示式也這樣做。
無效常數表示式的早期檢測。MySQL快速檢測某些SELECT語句是不可能的並且不返回行。
如果你不使用GROUP BY或分組函式(COUNT()、MIN()……),HAVING與WHERE合併。
為每個子聯結(sub join),構造一個更簡單的WHERE以得到一個更快的WHERE計算並且也儘快跳過記錄。
所有常數的表在查詢中的在其他任何表之前被讀出。
一個常數的表是:
一個空表或一個有1行的表。
與在一個UNIQUE索引、或一個PRIMARY KEY的WHERE子句一起使用的表,這裡所有的索引部分使用一個常數表示式並且索引部分被定義為NOT NULL。
所有下列的表用作常數表:
mysql> SELECT * FROM t WHERE primary_key=1;
mysql> SELECT * FROM t1,t2 WHERE t1.primary_key=1 AND t2.primary_key=t1.id;
對聯結表的最好聯結組合是透過嘗試所有可能性來找到:(。如果所有在ORDER BY和GROUP BY的列來自同一個表,那麼當聯結時,該表首先被選中。
如果你使用SQL_SMALL_RESULT,MySQL將使用一個在記憶體中的表。
如果有一個ORDER BY子句和一個不同的GROUP BY子句,或如果ORDER BY或GROUP BY包含不是來自聯結佇列中的第一個表的其他表的列,建立一個臨時表。
因為DISTINCT被變換到在所有的列上的一個GROUP BY,DISTINCT與ORDER BY結合也將在許多情況下需要一張臨時表。
每個表的索引被查詢並且使用跨越少於30% 的行的索引。如果這樣的索引沒能找到,將使用一個快速的表掃描。
在一些情況下,MySQL能從索引中讀出行,甚至不用查詢資料檔案。如果索引使用的所有列是數字的,那麼只有索引樹被用來解答查詢。
在每個記錄被輸出前,那些不匹配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;
下列查詢僅使用索引樹就可解決(假設索引列是數字的):
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實施了許多最佳化,但我沒時間全部測試.
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
索引使用的常數表示式僅計算一次。
在一個單個表上的沒有一個WHERE的COUNT(*)直接從表中檢索資訊。當僅使用一個表時,對任何NOT NULL表示式也這樣做。
無效常數表示式的早期檢測。MySQL快速檢測某些SELECT語句是不可能的並且不返回行。
如果你不使用GROUP BY或分組函式(COUNT()、MIN()……),HAVING與WHERE合併。
為每個子聯結(sub join),構造一個更簡單的WHERE以得到一個更快的WHERE計算並且也儘快跳過記錄。
所有常數的表在查詢中的在其他任何表之前被讀出。
一個常數的表是:
一個空表或一個有1行的表。
與在一個UNIQUE索引、或一個PRIMARY KEY的WHERE子句一起使用的表,這裡所有的索引部分使用一個常數表示式並且索引部分被定義為NOT NULL。
所有下列的表用作常數表:
mysql> SELECT * FROM t WHERE primary_key=1;
mysql> SELECT * FROM t1,t2 WHERE t1.primary_key=1 AND t2.primary_key=t1.id;
對聯結表的最好聯結組合是透過嘗試所有可能性來找到:(。如果所有在ORDER BY和GROUP BY的列來自同一個表,那麼當聯結時,該表首先被選中。
如果你使用SQL_SMALL_RESULT,MySQL將使用一個在記憶體中的表。
如果有一個ORDER BY子句和一個不同的GROUP BY子句,或如果ORDER BY或GROUP BY包含不是來自聯結佇列中的第一個表的其他表的列,建立一個臨時表。
因為DISTINCT被變換到在所有的列上的一個GROUP BY,DISTINCT與ORDER BY結合也將在許多情況下需要一張臨時表。
每個表的索引被查詢並且使用跨越少於30% 的行的索引。如果這樣的索引沒能找到,將使用一個快速的表掃描。
在一些情況下,MySQL能從索引中讀出行,甚至不用查詢資料檔案。如果索引使用的所有列是數字的,那麼只有索引樹被用來解答查詢。
在每個記錄被輸出前,那些不匹配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;
下列查詢僅使用索引樹就可解決(假設索引列是數字的):
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,...
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/10617731/viewspace-961903/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- MySQL最佳化WHERE子句(轉)MySql
- 詳解MySQL中WHERE子句的用法MySql
- MySQL-WHERE子句 以及 邏輯操作符MySql
- SQLite中的WHERE子句SQLite
- SQL中 where 子句和having子句中的區別SQL
- MyBatis中的<where>標籤和where子句的區別MyBatis
- MySQL 4.1.0 中文參考手冊 --- 6.3 用於 SELECT 和 WHERE 子句的函式 (1) (轉)MySql函式
- ORACLE SQL開發where子句之case-whenOracleSQL
- truncate和不帶where子句的delete, 以及drop區別delete
- 【實驗】where子句的解析順序及執行效率
- Oracle中where子句中條件的物理執行順序Oracle
- SQL優化] 避免在WHERE子句中對列使用函式SQL優化函式
- MySQL where 運算子MySql
- 【警惕】強化關聯更新時where子句範圍限定意識
- 怎樣使MySQL安全以對抗解密高手(轉)MySql解密
- MySQL的最佳化 (轉)MySql
- MySQL-where 1=1MySql
- 怎樣使MySQL更安全以對抗解密高手(轉)MySql解密
- MySQL UDF 在 in ( subquery where ) bugMySql
- MySQL中BETWEEN子句的用法詳解MySql
- MySQL join語句怎麼最佳化?MySql
- 帶你讀 MySQL 原始碼:where 條件怎麼過濾記錄?MySql原始碼
- 【Mysql 學習】Mysql 怎樣使用記憶體MySql記憶體
- 轉行學IT怎麼樣?
- 怎樣打敗微軟 (轉)微軟
- 怎樣手工宣告API (轉)API
- 怎樣捕捉 Delete 鍵 (轉)delete
- 非常好的MySQL最佳化(轉)MySql
- MySQL核心技術之“WHERE條件”MySql
- hibernate 中怎樣連線MYSQL??MySql
- 【北亞資料庫資料恢復】使用delete未加where子句刪除全表資料的Mysql資料庫資料恢復資料庫資料恢復deleteMySql
- 最佳化安裝 mysql+apache+php(轉)MySqlApachePHP
- 怎樣讓程式延時 (轉)
- 淺談怎樣在UNIX系統下安裝MySQL資料庫(轉)MySql資料庫
- Oracle行列轉換及pivot子句的用法Oracle
- (轉載)ORACLE MODEL子句學習筆記Oracle筆記
- MySQL之Where和Having的區別MySql
- php資料庫之mysql(where、orderBy、Update)PHP資料庫MySql