SQL 語句性能優(yōu)化策略

  1. 盡量避免全表掃描,首先考慮在where和order by涉及的列上簡歷索引
  2. 避免在where子句中對字段進行null判斷,所以最好在創(chuàng)建表的時候給每個字段都加默認值
  3. 避免在where中使用 != 或 <> ,MYSQL只有對以下操作符才使用索引: < , <= , = , > , >= , BETWEEN , IN 以及某些時候的LIKE
  4. 避免在where中使用or权烧,用了or會導致引擎放棄使用索引而使用全表掃描, 可以使用UNION 替代or
  5. in和not in也要慎用需纳,否則會導致全表掃描骗随,對于連續(xù)的數(shù)值蛤织,能用between就不要用in了:Select id from t where num between 1 and 3
  6. LIKE只有在 “abc%”的情況下才會用到索引,“%abc%”會導致全表掃描
  7. where中避免使用表達式操作
  8. 很多時候用exists代替in是一個好的選擇:select num from a where num in(select num from b)鸿染。用下面的語句替換:select num from a where exists(select 1 from b where num=a.num)
  9. 索引固然可以提高相應(yīng)的select的效率指蚜,但同時也降低了insert及update的效率,因為insert或update時有可能會重建索引涨椒,所以怎樣建索引需要慎重考慮摊鸡,視具體情況而定。一個表的索引數(shù)最好不要超過6個蚕冬,若太多則應(yīng)考慮一些不常使用到的列上建的索引是否有必要
  10. 盡量使用數(shù)字型字段免猾,若只含數(shù)值信息的字段盡量不要設(shè)計為字符型,這會降低查詢和連接的性能囤热,并會增加存儲開銷
  11. 最好不要使用 ”*“ 返回所有:select from t 猎提,用具體的字段列表代替 “*” ,不要返回用不到的任何字段
  12. 使用表的別名(Alias):當在SQL語句中連接多個表時旁蔼,請使用表的別名并把別名前綴于每個Column上锨苏。這樣一來,就可以減少解析的時間并減少那些由Column歧義引起的語法錯誤
  13. 在IN后面值的列表中棺聊,將出現(xiàn)最頻繁的值放在最前面伞租,出現(xiàn)得最少的放在最后面,減少判斷的次數(shù)
  14. 盡量使用exists代替select count(1)來判斷是否存在記錄限佩,count函數(shù)只有在統(tǒng)計表中所有行數(shù)時使用葵诈,而且count(1)比count(*)更有效率
  15. 盡量使用“>=”,不要使用“>”
  16. 下列SQL條件語句中的列都建有恰當?shù)乃饕钔珗?zhí)行速度卻非常慢:
SELECT * FROM record WHERE substrINg(card_no,1,4)=’5378’ (13秒) 

SELECT * FROM record WHERE amount/30< 1000 (11秒) 

SELECT * FROM record WHERE convert(char(10),date,112)=’19991201’ (10秒) 

分析:

WHERE子句中對列的任何操作結(jié)果都是在SQL運行時逐列計算得到的驯击,因此它不得不進行表搜索,而沒有使用該列上面的索引耐亏。

如果這些結(jié)果在查詢編譯時就能得到徊都,那么就可以被SQL優(yōu)化器優(yōu)化,使用索引广辰,避免表搜索暇矫,因此將SQL重寫成下面這樣:

SELECT * FROM record WHERE card_no like ‘5378%’ (< 1秒) 

SELECT * FROM record WHERE amount< 1000*30 (< 1秒) 

SELECT * FROM record WHERE date= ‘1999/12/01’ (< 1秒)
  1. 提高GROUP BY語句的效率,可以通過將不需要的記錄在GROUP BY之前過濾掉择吊。下面兩個查詢返回相同結(jié)果李根,但第二個明顯就快了許多
-- 低效
SELECT JOB , AVG(SAL) FROM EMP GROUP BY JOB HAVING JOB =’PRESIDENT’ OR JOB =’MANAGER’ 
-- 高效
SELECT JOB , AVG(SAL) FROM EMP WHERE JOB =’PRESIDENT’ OR JOB =’MANAGER’ GROUP BY JOB
  1. SQL語句用大寫,因為Oracle總是先解析SQL語句几睛,把小寫的字母轉(zhuǎn)換成大寫的再執(zhí)行

  2. 別名的使用房轿,別名是大型數(shù)據(jù)庫的應(yīng)用技巧,就是表名、列名在查詢中以一個字母為別名囱持,查詢速度要比建連接表快1.5倍

  3. 索引創(chuàng)建規(guī)則:

    • 表的主鍵夯接、外鍵必須有索引;
    • 數(shù)據(jù)量超過300的表應(yīng)該有索引纷妆;
    • 經(jīng)常與其他表進行連接的表盔几,在連接字段上應(yīng)該建立索引;
    • 經(jīng)常出現(xiàn)在Where子句中的字段掩幢,特別是大表的字段逊拍,應(yīng)該建立索引;
    • 索引應(yīng)該建在選擇性高的字段上际邻;
    • 索引應(yīng)該建在小字段上芯丧,對于大的文本字段甚至超長字段,不要建索引世曾;
    • 復合索引的建立需要進行仔細分析缨恒,盡量考慮用單字段索引代替;
    • 正確選擇復合索引中的主列字段度硝,一般是選擇性較好的字段肿轨;
    • 復合索引的幾個字段是否經(jīng)常同時以AND方式出現(xiàn)在Where子句中寿冕?單字段查詢是否極少甚至沒有蕊程?如果是,則可以建立復合索引驼唱;否則考慮單字段索引藻茂;
    • 如果復合索引中包含的字段經(jīng)常單獨出現(xiàn)在Where子句中,則分解為多個單字段索引玫恳;
    • 如果復合索引所包含的字段超過3個辨赐,那么仔細考慮其必要性,考慮減少復合的字段京办;
    • 如果既有單字段索引掀序,又有這幾個字段上的復合索引,一般可以刪除復合索引惭婿;
    • 頻繁進行數(shù)據(jù)操作的表不恭,不要建立太多的索引;
    • 刪除無用的索引财饥,避免對執(zhí)行計劃造成負面影響换吧;
    • 表上建立的每個索引都會增加存儲開銷,索引對于插入钥星、刪除沾瓦、更新操作也會增加處理上的開銷。另外,過多的復合索引贯莺,在有單字段索引的情況下风喇,一般都是沒有存在價值的;相反乖篷,還會降低數(shù)據(jù)增加刪除時的性能响驴,特別是對頻繁更新的表來說,負面影響更大
    • 盡量不要對數(shù)據(jù)庫中某個含有大量重復的值的字段建立索引
  4. 我們應(yīng)該為數(shù)據(jù)庫里的每張表都設(shè)置一個ID做為其主鍵撕蔼,而且最好的是一個INT型的(推薦使用UNSIGNED)豁鲤,并設(shè)置上自動增加的AUTO_INCREMENT標志

  5. 當只要一行數(shù)據(jù)時使用LIMIT 1

  6. 優(yōu)化表的數(shù)據(jù)類型,選擇合適的數(shù)據(jù)類型:

    原則:更小通常更好鲸沮,簡單就好琳骡,所有字段都得有默認值,盡量避免null

    例如:數(shù)據(jù)庫表設(shè)計時候更小的占磁盤空間盡可能使用更小的整數(shù)類型讼溺。(mediumint就比int更合適)

    比如時間字段:datetime和timestamp楣号,datetime占用8個字節(jié),而timestamp占用4個字節(jié)怒坯,只用了一半炫狱,而timestamp表示的范圍是1970—2037適合做更新時間

    MySQL可以很好的支持大數(shù)據(jù)量的存取,但是一般說來剔猿,數(shù)據(jù)庫中的表越小视译,在它上面執(zhí)行的查詢也就會越快

    因此,在創(chuàng)建表的時候归敬,為了獲得更好的性能酷含,我們可以將表中字段的寬度設(shè)得盡可能小

    例如:在定義郵政編碼這個字段時,如果將其設(shè)置為CHAR(255)汪茧,顯然給數(shù)據(jù)庫增加了不必要的空間椅亚。甚至使用VARCHAR這種類型也是多余的,因為CHAR(6)就可以很好的完成任務(wù)了

    同樣的舱污,如果可以的話呀舔,我們應(yīng)該使用MEDIUMINT而不是BIGIN來定義整型字段,應(yīng)該盡量把字段設(shè)置為NOT NULL扩灯,這樣在將來執(zhí)行查詢的時候媚赖,數(shù)據(jù)庫不用去比較NULL值

    對于某些文本字段,例如“省份”或者“性別”驴剔,我們可以將它們定義為ENUM類型省古。因為在MySQL中,ENUM類型被當作數(shù)值型數(shù)據(jù)來處理丧失,而數(shù)值型數(shù)據(jù)被處理起來的速度要比文本類型快得多這樣豺妓,我們又可以提高數(shù)據(jù)庫的性能

  1. 字符串數(shù)據(jù)類型:char,varchar,text選擇區(qū)別
  2. 任何對列的操作都將導致表掃描琳拭,它包括數(shù)據(jù)庫函數(shù)训堆、計算表達式等等,查詢時要盡可能將操作移至等號右邊
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末白嘁,一起剝皮案震驚了整個濱河市坑鱼,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌絮缅,老刑警劉巖鲁沥,帶你破解...
    沈念sama閱讀 212,718評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異耕魄,居然都是意外死亡画恰,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,683評論 3 385
  • 文/潘曉璐 我一進店門吸奴,熙熙樓的掌柜王于貴愁眉苦臉地迎上來允扇,“玉大人,你說我怎么就攤上這事则奥】既螅” “怎么了?”我有些...
    開封第一講書人閱讀 158,207評論 0 348
  • 文/不壞的土叔 我叫張陵读处,是天一觀的道長糊治。 經(jīng)常有香客問我,道長档泽,這世上最難降的妖魔是什么俊戳? 我笑而不...
    開封第一講書人閱讀 56,755評論 1 284
  • 正文 為了忘掉前任揖赴,我火速辦了婚禮馆匿,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘燥滑。我一直安慰自己渐北,他們只是感情好,可當我...
    茶點故事閱讀 65,862評論 6 386
  • 文/花漫 我一把揭開白布铭拧。 她就那樣靜靜地躺著赃蛛,像睡著了一般。 火紅的嫁衣襯著肌膚如雪搀菩。 梳的紋絲不亂的頭發(fā)上呕臂,一...
    開封第一講書人閱讀 50,050評論 1 291
  • 那天,我揣著相機與錄音肪跋,去河邊找鬼歧蒋。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的谜洽。 我是一名探鬼主播萝映,決...
    沈念sama閱讀 39,136評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼阐虚!你這毒婦竟也來了序臂?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,882評論 0 268
  • 序言:老撾萬榮一對情侶失蹤实束,失蹤者是張志新(化名)和其女友劉穎奥秆,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體咸灿,經(jīng)...
    沈念sama閱讀 44,330評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡吭练,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,651評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了析显。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片鲫咽。...
    茶點故事閱讀 38,789評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖谷异,靈堂內(nèi)的尸體忽然破棺而出分尸,到底是詐尸還是另有隱情,我是刑警寧澤歹嘹,帶...
    沈念sama閱讀 34,477評論 4 333
  • 正文 年R本政府宣布箩绍,位于F島的核電站,受9級特大地震影響尺上,放射性物質(zhì)發(fā)生泄漏材蛛。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 40,135評論 3 317
  • 文/蒙蒙 一怎抛、第九天 我趴在偏房一處隱蔽的房頂上張望卑吭。 院中可真熱鬧,春花似錦马绝、人聲如沸豆赏。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,864評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽掷邦。三九已至,卻和暖如春椭赋,著一層夾襖步出監(jiān)牢的瞬間抚岗,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,099評論 1 267
  • 我被黑心中介騙來泰國打工哪怔, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留宣蔚,地道東北人廷痘。 一個月前我還...
    沈念sama閱讀 46,598評論 2 362
  • 正文 我出身青樓,卻偏偏與公主長得像件已,于是被迫代替她去往敵國和親笋额。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 43,697評論 2 351