我們要做到不但會寫SQL,還要做到寫出性能優良的SQL語句。 1.使用表的別名(Alias): 當在SQL語句中連接多個表時, 請使用表的別名並把別名首碼於每個Column上。這樣一來,就可以減少解析的時間並減少那些由Column歧義引起的語法錯誤。 2.表的索引: 索引是表的一個概念部分,用來提高 ...
我們要做到不但會寫SQL,還要做到寫出性能優良的SQL語句。
1.使用表的別名(Alias):
當在SQL語句中連接多個表時, 請使用表的別名並把別名首碼於每個Column上。這樣一來,就可以減少解析的時間並減少那些由Column歧義引起的語法錯誤。
2.表的索引:
索引是表的一個概念部分,用來提高檢索數據的效率,Oracle使用了一個複雜的自平衡B-tree結構。通常,通過索引查詢數據比全表掃描要快。當 Oracle找出執行查詢和Update語句的最佳路徑時, Oracle優化器將使用索引。同樣在聯結多個表時使用索引也可以提高效率。另一個使用索引的好處是,它提供了主鍵(primary key)的唯一性驗證。那些LONG或LONG RAW數據類型, 你可以索引幾乎所有的列。通常, 在大型表中使用索引特別有效. 當然,你也會發現, 在掃描小表時,使用索引同樣能提高效率。雖然使用索引能得到查詢效率的提高,但是我們也必須註意到它的代價。索引需要空間來存儲,也需要定期維護, 每當有記錄在表中增減或索引列被修改時, 索引本身也會被修改。這意味著每條記錄的INSERT,DELETE , UPDATE將為此多付出4、 5次的磁碟I/O 。因為索引需要額外的存儲空間和處理,那些不必要的索引反而會使查詢反應時間變慢。定期的重構索引是有必要的:
ALTER INDEX <INDEXNAME> REBUILD <TABLESPACENAME>
3.避免在索引列上使用IS NULL和IS NOT NULL:
避免在索引中使用任何可以為空的列,Oracle將無法使用該索引。對於單列索引,如果列包含空值,索引中將不存在此記錄。對於複合索引,如果每個列都為 空,索引中同樣不存在此記錄。如果至少有一個列不為空,則記錄存在於索引中。舉例:如果唯一性索引建立在表的A列和B列上,並且表中存在一條記錄的A,B 值為(123,null), Oracle將不接受下一條具有相同A,B值(123,null)的記錄(插入)。 然而如果所有的索引列都為空,Oracle將認為整個鍵值為空而空不等於空。因此你可以插入1000 條具有相同鍵值的記錄,當然它們都是空! 因為空值不存在於索引列中,所以WHERE子句中對索引列進行空值比較將使ORACLE停用該索引。
4.需要當心的WHERE子句:
某些SELECT 語句中的WHERE子句不使用索引。這裡有一些例子:
(1)‘!=' 將不使用索引。記住, 索引只能告訴你什麼存在於表中, 而不能告訴你什麼不存在於表中。
(2)‘||'是字元連接函數。就象其他函數那樣, 停用了索引。
(3)‘+'是數學函數。就象其他數學函數那樣, 停用了索引。
(4)相同的索引列不能互相比較,這將會啟用全表掃描。
(5)a. 如果檢索數據量超過30%的表中記錄數,使用索引將沒有顯著的效率提高。
b. 在特定情況下,使用索引也許會比全表掃描慢,但這是同一個數量級上的區別。而通常情況下,使用索引比全表掃描要塊幾倍乃至幾千倍!
5.SQL語句用大寫的;因為Oracle總是先解析SQL語句,把小寫的字母轉換成大寫的再執行。
實際案例分析:拆分大的 DELETE 或INSERT 語句,批量提交SQL語句
如果你需要在一個線上的網站上去執行一個大的 DELETE 或 INSERT 查詢,你需要非常小心,要避免你的操作讓你的整個網站停止相應。因為這兩個操作是會鎖表的,表一鎖住了,別的操作都進不來了。
Apache 會有很多的子進程或線程。所以,其工作起來相當有效率,而我們的伺服器也不希望有太多的子進程,線程和資料庫鏈接,這是極大的占伺服器資源的事情,尤其是記憶體。
如果你把你的表鎖上一段時間,比如30秒鐘,那麼對於一個有很高訪問量的站點來說,這30秒所積累的訪問進程/線程,資料庫鏈接,打開的文件數,可能不僅僅會讓你的WEB服務崩潰,還可能會讓你的整台伺服器馬上掛了。
所以,如果你有一個大的處理,你一定把其拆分,使用 LIMIT oracle(rownum),sqlserver(top)條件是一個好的方法。下麵是一個mysql示例:
while(1){ //每次只做1000條 mysql_query(“delete from logs where log_date <= ’2012-11-01’ limit 1000”); if(mysql_affected_rows() == 0){
//刪除完成,退出! break; } //每次暫停一段時間,釋放表讓其他進程/線程訪問。 usleep(50000) }
好了,到這裡就寫完了。我知道還有很多很多沒有寫到的,還請大家補充。讓我們一起學習,一起進步吧!