當(dāng)前位置:首頁(yè) > IT技術(shù) > 數(shù)據(jù)庫(kù) > 正文

數(shù)據(jù)庫(kù)sql優(yōu)化總結(jié)之1-百萬(wàn)級(jí)數(shù)據(jù)庫(kù)優(yōu)化方案+案例分析
2021-12-13 17:45:55

目錄

項(xiàng)目背景

1.對(duì)查詢進(jìn)行優(yōu)化,要盡量避免全表掃描,首先應(yīng)考慮在 where 及 order by 涉及的列上建立索引。

案例分析:

2、應(yīng)盡量避免在 where 子句中對(duì)字段進(jìn)行 null 值判斷,否則將導(dǎo)致引擎放棄使用索引而進(jìn)行全表掃描

案例分析:

3、應(yīng)盡量避免在 where 子句中使用 != 或 <> 操作符,否則將引擎放棄使用索引而進(jìn)行全表掃描。

案例分析:

4.應(yīng)盡量避免在 where 子句中使用 or 來(lái)連接條件,如果一個(gè)字段有索引,一個(gè)字段沒(méi)有索引,將導(dǎo)致引擎放棄使用索引而進(jìn)行全表掃描

案例分析:

5.in 和 not in 也要慎用,否則會(huì)導(dǎo)致全表掃描

案例分析

案例分析2:

案例分析3:

6、like模糊全匹配也將導(dǎo)致全表掃描

案例分析


項(xiàng)目背景

有三張百萬(wàn)級(jí)數(shù)據(jù)表

知識(shí)點(diǎn)表(ex_subject_point)9,316條數(shù)據(jù)

試題表(ex_question_junior)2,159,519條數(shù)據(jù) 有45個(gè)字段

知識(shí)點(diǎn)試題關(guān)系表(ex_question_r_knowledge)3,156,155條數(shù)據(jù)

測(cè)試數(shù)據(jù)庫(kù)為:mysql (5.7)

?

1.對(duì)查詢進(jìn)行優(yōu)化,要盡量避免全表掃描,首先應(yīng)考慮在 where 及 order by 涉及的列上建立索引。

案例分析:

SELECT ex_question_junior.QUESTION_ID 
FROM ex_question_junior 
WHERE ex_question_junior.GRADE_ID=1

    執(zhí)行時(shí)間:17.609s (多次執(zhí)行,在17s左右徘徊)

  優(yōu)化后:給GRADE_ID字段添加索引后

  執(zhí)行時(shí)間為:11.377s(多次執(zhí)行,在11s左右徘徊)

  備注:我們一般在什么字段上建索引?

  這是一個(gè)非常復(fù)雜的話題,需要對(duì)業(yè)務(wù)及數(shù)據(jù)充分分析后再能得出結(jié)果。主鍵及外鍵通常都要有索引,其它需要建索引的字段應(yīng)滿足以下條件:

    a、字段出現(xiàn)在查詢條件中,并且查詢條件可以使用索引;

    b、語(yǔ)句執(zhí)行頻率高,一天會(huì)有幾千次以上;

    c、通過(guò)字段條件可篩選的記錄集很小,那數(shù)據(jù)篩選比例是多少才適合?

  這個(gè)沒(méi)有固定值,需要根據(jù)表數(shù)據(jù)量來(lái)評(píng)估,以下是經(jīng)驗(yàn)公式,可用于快速評(píng)估:

  小表(記錄數(shù)小于10000行的表):篩選比例<10%;

  大表:(篩選返回記錄數(shù))<(表總記錄數(shù)*單條記錄長(zhǎng)度)/10000/16

  單條記錄長(zhǎng)度≈字段平均內(nèi)容長(zhǎng)度之和+字段數(shù)*2

  以下是一些字段是否需要建B-TREE索引的經(jīng)驗(yàn)分類:

數(shù)據(jù)庫(kù)sql優(yōu)化總結(jié)之1-百萬(wàn)級(jí)數(shù)據(jù)庫(kù)優(yōu)化方案+案例分析_數(shù)據(jù)庫(kù)

2、應(yīng)盡量避免在 where 子句中對(duì)字段進(jìn)行 null 值判斷,否則將導(dǎo)致引擎放棄使用索引而進(jìn)行全表掃描

select id from t where num is null

最好不要給數(shù)據(jù)庫(kù)留NULL,盡可能的使用?NOT NULL填充數(shù)據(jù)庫(kù).

備注、描述、評(píng)論之類的可以設(shè)置為?NULL,其他的,最好不要使用NULL。

不要以為?NULL?不需要空間,比如:char(100)?型,在字段建立時(shí),空間就固定了,?不管是否插入值(NULL也包含在內(nèi)),都是占用?100個(gè)字符的空間的,如果是varchar這樣的變長(zhǎng)字段,?null?不占用空間。

可以在num上設(shè)置默認(rèn)值0,確保表中num列沒(méi)有null值,然后這樣查詢:

select id from t where num = 0

案例分析:

在mysql數(shù)據(jù)庫(kù)中對(duì)字段進(jìn)行null值判斷,是不會(huì)放棄使用索引而進(jìn)行全表掃描的。

SELECT ex_question_junior.QUESTION_ID
FROM ex_question_junior 
WHERE IS_USE is NULL

?

執(zhí)行時(shí)間是:11.729s

?

SELECT ex_question_junior.QUESTION_ID
FROM ex_question_junior 
WHERE IS_USE =0

執(zhí)行時(shí)間是12.253s

時(shí)間幾乎一樣。

3、應(yīng)盡量避免在 where 子句中使用 != 或 <> 操作符,否則將引擎放棄使用索引而進(jìn)行全表掃描。

案例分析:

在mysql數(shù)據(jù)庫(kù)中where 子句中使用 != 或 <> 操作符,引擎不會(huì)放棄使用索引。

EXPLAIN
SELECT ex_question_junior.QUESTION_ID
FROM ex_question_junior 
WHERE ex_question_junior.GRADE_ID !=15

數(shù)據(jù)庫(kù)sql優(yōu)化總結(jié)之1-百萬(wàn)級(jí)數(shù)據(jù)庫(kù)優(yōu)化方案+案例分析_案例分析_02

執(zhí)行時(shí)間是:17.579s

數(shù)據(jù)庫(kù)sql優(yōu)化總結(jié)之1-百萬(wàn)級(jí)數(shù)據(jù)庫(kù)優(yōu)化方案+案例分析_百萬(wàn)數(shù)據(jù)_03

執(zhí)行時(shí)間是:16.966s

4.應(yīng)盡量避免在 where 子句中使用 or 來(lái)連接條件,如果一個(gè)字段有索引,一個(gè)字段沒(méi)有索引,將導(dǎo)致引擎放棄使用索引而進(jìn)行全表掃描

案例分析:

GRADE_ID字段有索引,QUESTION_TYPE沒(méi)索引

數(shù)據(jù)庫(kù)sql優(yōu)化總結(jié)之1-百萬(wàn)級(jí)數(shù)據(jù)庫(kù)優(yōu)化方案+案例分析_數(shù)據(jù)庫(kù)_04

執(zhí)行時(shí)間是:11.661s

優(yōu)化方案:

通過(guò)union all 方式,把有索引字段和非索引字段分開(kāi)。索引字段就有效果了

數(shù)據(jù)庫(kù)sql優(yōu)化總結(jié)之1-百萬(wàn)級(jí)數(shù)據(jù)庫(kù)優(yōu)化方案+案例分析_百萬(wàn)數(shù)據(jù)_05

執(zhí)行時(shí)間是:11.811s

但是,非索引字段依然查詢速度會(huì)很慢,所以查詢條件,能加索引的盡量加索引

5.in 和 not in 也要慎用,否則會(huì)導(dǎo)致全表掃描

案例分析

注:在mysql數(shù)據(jù)庫(kù)中where 子句中對(duì)索引字段使用 in 和 not in操作符,引擎不會(huì)放棄使用索引。

數(shù)據(jù)庫(kù)sql優(yōu)化總結(jié)之1-百萬(wàn)級(jí)數(shù)據(jù)庫(kù)優(yōu)化方案+案例分析_優(yōu)化_06

注:在mysql數(shù)據(jù)庫(kù)中where 子句中對(duì)不是索引字段使用 in 和 not in操作符,會(huì)導(dǎo)致全表掃描。

數(shù)據(jù)庫(kù)sql優(yōu)化總結(jié)之1-百萬(wàn)級(jí)數(shù)據(jù)庫(kù)優(yōu)化方案+案例分析_優(yōu)化_07

案例分析2:

用between和in的區(qū)別

SELECT ex_question_junior.QUESTION_ID
FROM ex_question_junior
WHERE ex_question_junior.QUESTION_TYPE  IN(1,2,3,4)

執(zhí)行時(shí)間為1.082s

SELECT ex_question_junior.QUESTION_ID
FROM ex_question_junior
WHERE ex_question_junior.QUESTION_TYPE between 1 and 4

?

執(zhí)行時(shí)間為0.924s

?

時(shí)間上是相差不多的

案例分析3:

用exists 和 in區(qū)別:結(jié)論

用exists 和 in區(qū)別:結(jié)論

1. in()適合B表比A表數(shù)據(jù)大的情況(A<B)

select * from A

where id in(select id from B)

2. exists()適合B表比A表數(shù)據(jù)小的情況(A>B)

  select * from A

  where exists(

  select 1 from B where B.id = A.id

  )

3.當(dāng)A表數(shù)據(jù)與B表數(shù)據(jù)一樣大時(shí),in與exists效率差不多,可任選一個(gè)使用.語(yǔ)法

select * from A

where id in(select id from B)

ex_question_r_knowledge表數(shù)據(jù)量大,ex_subject_point表數(shù)據(jù)量小

****************************************************************************

ex_question_r_knowledge(A)表數(shù)據(jù)量大,ex_subject_point表數(shù)據(jù)量小(B)(A>B)

用exists適合

SELECT *
FROM ex_question_r_knowledge
WHERE ex_question_r_knowledge.SUBJECT_POINT_ID IN
(
    SELECT ex_subject_point.SUBJECT_POINT_ID
    FROM ex_subject_point
    WHERE ex_subject_point.SUBJECT_ID=7
)

?

SELECT *
FROM ex_question_r_knowledge
WHERE  exists
(
    SELECT 1
    FROM ex_subject_point
    WHERE ex_subject_point.SUBJECT_ID=7
    AND ex_subject_point.SUBJECT_POINT_ID = ex_question_r_knowledge.SUBJECT_POINT_ID
)

?

執(zhí)行時(shí)間是:13.537s

*************************************************************************

ex_subject_point表數(shù)據(jù)量小(A),ex_question_r_knowledge(B)表數(shù)據(jù)量大(A<B)

用in適合

SELECT *
 FROM ex_subject_point 
WHERE
 ex_subject_point.SUBJECT_POINT_ID IN( SELECT 
ex_question_r_knowledge.SUBJECT_POINT_ID FROM 
ex_question_r_knowledge WHERE 
ex_question_r_knowledge.GRADE_TYPE=2 )

?

?

SELECT * FROM ex_subject_point WHERE
 ex_subject_point.SUBJECT_POINT_ID IN( SELECT 
ex_question_r_knowledge.SUBJECT_POINT_ID FROM 
ex_question_r_knowledge WHERE 
ex_question_r_knowledge.GRADE_TYPE=2 )

執(zhí)行時(shí)間是:1.554s

SELECT *
    FROM ex_subject_point
    WHERE  exists(
    SELECT ex_question_r_knowledge.SUBJECT_POINT_ID
    FROM ex_question_r_knowledge
    WHERE ex_question_r_knowledge.GRADE_TYPE=2
    AND ex_question_r_knowledge.SUBJECT_POINT_ID= ex_subject_point.SUBJECT_POINT_ID
)

執(zhí)行時(shí)間是:11.978s

6、like模糊全匹配也將導(dǎo)致全表掃描

案例分析

EXPLAIN
 SELECT * 
FROM ex_subject_point 
WHERE ex_subject_point.path like "%/11/%"

?

數(shù)據(jù)庫(kù)sql優(yōu)化總結(jié)之1-百萬(wàn)級(jí)數(shù)據(jù)庫(kù)優(yōu)化方案+案例分析_數(shù)據(jù)庫(kù)_08

若要提高效率,可以考慮全文檢索。lucene了解一下?;蛘咂渌梢蕴峁┤乃饕膎osql數(shù)據(jù)庫(kù),比如tt server或MongoDB

還會(huì)陸續(xù)更新,還有幾個(gè)小節(jié)。

?

昨天晚上突發(fā)奇想,like 模糊全匹配,會(huì)導(dǎo)致全表掃描,那模糊后匹配和模糊前匹配也會(huì)是全表掃描嗎?

今天開(kāi)電腦,做了下測(cè)試。結(jié)果如下:

like模糊后匹配,不會(huì)導(dǎo)致全表掃描

數(shù)據(jù)庫(kù)sql優(yōu)化總結(jié)之1-百萬(wàn)級(jí)數(shù)據(jù)庫(kù)優(yōu)化方案+案例分析_優(yōu)化_09

like模糊前匹配,會(huì)導(dǎo)致全表掃描

數(shù)據(jù)庫(kù)sql優(yōu)化總結(jié)之1-百萬(wàn)級(jí)數(shù)據(jù)庫(kù)優(yōu)化方案+案例分析_優(yōu)化_10

MY SQL的原理就是這樣的,LIKE模糊全匹配會(huì)導(dǎo)致索引失效,進(jìn)行全表掃描;LIKE模糊前匹配也會(huì)導(dǎo)致索引失效,進(jìn)行全表掃描;但是LIKE模糊后匹配,索引就會(huì)有效果。

?

參考:

?

https://mp.weixin.qq.com/s?__biz=MzIxMjg4NDU1NA==&mid=2247483684&idx=1&sn=f5abc60e696b2063e43cd9ccb40df101&chksm=97be0c01a0c98517029ff9aa280b398ab5c81fa1fcfe0e746222a3bfe75396d9eea1e249af38&mpshare=1&scene=1&srcid=0606XGHeBS4RBZloVv786wBY#rd

?

***************************************************************************

作者:小虛竹
歡迎任何形式的轉(zhuǎn)載,但請(qǐng)務(wù)必注明出處。
限于本人水平,如果文章和代碼有表述不當(dāng)之處,還請(qǐng)不吝賜教。

?

我不是個(gè)偉大的程序員,我只是個(gè)有著一些優(yōu)秀習(xí)慣的好程序員而己

?

本文摘自 :https://blog.51cto.com/u

開(kāi)通會(huì)員,享受整站包年服務(wù)立即開(kāi)通 >