濮阳杆衣贸易有限公司

主頁(yè) > 知識(shí)庫(kù) > 如何利用分析函數(shù)改寫(xiě)范圍判斷自關(guān)聯(lián)查詢?cè)斀?/div>

如何利用分析函數(shù)改寫(xiě)范圍判斷自關(guān)聯(lián)查詢?cè)斀?/h1>

熱門(mén)標(biāo)簽:小程序智能電話機(jī)器人 怎么申請(qǐng)400熱線電話 河北便宜電銷機(jī)器人軟件 泗洪正規(guī)電話機(jī)器人找哪家 簡(jiǎn)單的智能語(yǔ)音電銷機(jī)器人 湖南保險(xiǎn)智能外呼系統(tǒng)產(chǎn)品介紹 南昌呼叫中心外呼系統(tǒng)哪家好 ai電話電話機(jī)器人 怎么去開(kāi)發(fā)一個(gè)電銷機(jī)器人

前言

最近碰到一個(gè)單條SQL運(yùn)行效率不佳導(dǎo)致數(shù)據(jù)庫(kù)整體運(yùn)行負(fù)載較高的問(wèn)題。

分析、定位數(shù)據(jù)庫(kù)的主要負(fù)載是這條語(yǔ)句引起的過(guò)程相對(duì)簡(jiǎn)單,通過(guò)AWR報(bào)告就可以比較容易的完成定位,這里就不贅述了。

現(xiàn)在直接看一下這個(gè)導(dǎo)致性能問(wèn)題的SQL語(yǔ)句,其對(duì)應(yīng)的SQL REPORT統(tǒng)計(jì)如下:

Stat Name Statement Total Per Execution % Snap Total
Elapsed Time (ms) 363,741 363,740.78 8 .42
CPU Time (ms) 362,770 362,770.00 8 .81
Executions 1    
Buffer Gets 756 756.00 0.00
Disk Reads 0 0.00 0.00
Parse Calls 1 1.00 0.01
Rows 50,825 50,825.00  
User I/O Wait Time (ms) 0  
Cluster Wait Time (ms) 0    
Application Wait Time (ms) 0    
Concurrency Wait Time (ms) 0    
Invalidations 0    
Version Count 1    
Sharable Mem(KB) 28    

從SQL的性能指標(biāo)上看,其單次執(zhí)行需要6分鐘左右,處理5萬(wàn)多條記錄,邏輯度只有756,主要消耗時(shí)間在CPU上。而這里就存在疑點(diǎn),邏輯讀如此之低,而CPU時(shí)間花費(fèi)又如此之高,那么這些CPU都消耗在哪里呢?當(dāng)然這個(gè)問(wèn)通過(guò)SQL的統(tǒng)計(jì)信息中是找不到答案的,我們下面關(guān)注SQL的執(zhí)行計(jì)劃:


Id Operation Name Rows Bytes TempSpc Cost (%CPU) Time
0 SELECT STATEMENT       1226 (100)  
1    SORT ORDER BY   49379 3375K 3888K 1226 (2) 00:00:05
2      HASH JOIN ANTI   49379 3375K 2272K 401 (3) 00:00:02
3        TABLE ACCESS FULL T_NUM 49379 1687K   88 (4) 00:00:01
4        TABLE ACCESS FULL T_NUM 49379 1687K   88 (4) 00:00:01

從執(zhí)行計(jì)劃看,Oracle選擇了HASH JOIN ANTI,JOIN的兩張表都是T_NUM,且都采用了全表掃描,并未選擇索引。僅靠執(zhí)行計(jì)劃也只等得到上面的結(jié)論,至于為什么不選擇索引,以及為什么執(zhí)行時(shí)間過(guò)長(zhǎng),還需要進(jìn)一步的分析。

將原SQL進(jìn)行簡(jiǎn)單脫密改寫(xiě)后, SQL文本類似如下:

SELECT BEGIN, END, ROWID, LENGTH(BEGIN)
FROM T_NUM A
WHERE NOT EXISTS (
SELECT 1
FROM T_NUM B
WHERE B.BEGIN = A.BEGIN
AND B.END >= A.END
AND B.ROWID != A.ROWID
AND LENGTH(B.BEGIN) = LENGTH(A.BEGIN));

如果分析SQL語(yǔ)句,會(huì)發(fā)現(xiàn)這是一個(gè)自關(guān)聯(lián)語(yǔ)句,在BEGIN字段長(zhǎng)度相等的前提下,想要找到哪些不存在BEGIN比當(dāng)前記錄BEGIN小且END比當(dāng)前記錄END大的記錄。

簡(jiǎn)單一點(diǎn)說(shuō),表中的記錄表示的是由BEGIN開(kāi)始到END截至的范圍,那么當(dāng)前想要獲取的結(jié)果是找出哪些沒(méi)有范圍所包含的范圍。需要注意的是,對(duì)于當(dāng)前的SQL邏輯,如果存在兩條范圍完全相同的記錄,那么最終這兩條記錄都會(huì)被舍棄。

業(yè)務(wù)的邏輯并不是特別復(fù)雜,但是要解決一條記錄與其他記錄進(jìn)行比較,多半采用的方法是自關(guān)聯(lián),而在這個(gè)自關(guān)聯(lián)中,既有大于等于又有小于等于,還有不等于,僅有的一個(gè)等于的關(guān)聯(lián)條件,來(lái)自范圍段BEGIN的長(zhǎng)度的比較。

顯而易見(jiàn)的是,如果是范圍段本身的比較,其選擇度一般還是不錯(cuò)的,但是如果只是比較其長(zhǎng)度,那么無(wú)疑容易產(chǎn)生大量的重復(fù),比如在這個(gè)例子中:

SQL> select length(begin), count(*) from t_num group by length(begin) order by 2 desc;

 

LENGTH(BEGIN) COUNT(*)

————- ———-

12  22096

11  9011

13  8999

14  8186

16   49

9   45

8   41

7   27

大量重復(fù)的數(shù)據(jù)出現(xiàn)在長(zhǎng)度為11到14的范圍上,在這種情況下,僅有的一個(gè)等值判斷條件LENGTH(BEGIN)是非常低效的,這時(shí)一條記錄根據(jù)這個(gè)等值條件會(huì)關(guān)聯(lián)到近萬(wàn)條記錄,設(shè)置關(guān)聯(lián)到兩萬(wàn)多條記錄,顯然大量的實(shí)踐消耗在低效的連接過(guò)程中。

再來(lái)看一下具體的SQL語(yǔ)句,會(huì)發(fā)現(xiàn)幾乎沒(méi)有辦法建立索引,因?yàn)長(zhǎng)ENGTH(BEGIN)的選擇度非常查,而其他的條件都是不等查詢,選擇度也不會(huì)好,即使建立索引,強(qiáng)制執(zhí)行選擇索引,效率也不會(huì)好。

那么如果想要繼續(xù)優(yōu)化這個(gè)SQL,就只剩下一個(gè)辦法,那就是SQL的改寫(xiě)。對(duì)于自關(guān)聯(lián)查詢而言,最佳的改寫(xiě)方法是利用分析函數(shù),其強(qiáng)大的行級(jí)處理能力,可以在一次掃描過(guò)程中獲得一條記錄與其他記錄的關(guān)系,從而消除了自關(guān)聯(lián)的必要性。

SQL改寫(xiě)結(jié)果如下:

SELECT BEGIN, OLDEND END, LENGTH(BEGIN)
FROM (
SELECT BEGIN, OLDEND, END, LENGTH(BEGIN), COUNT(*) OVER(PARTITION BY LENGTH(BEGIN), BEGIN, OLDEND) CN,
ROW_NUMBER() OVER(PARTITION BY LENGTH(BEGIN), END ORDER BY BEGIN) RN
FROM
(
SELECT BEGIN, END OLDEND, MAX(END) OVER(PARTITION BY LENGTH(BEGIN) ORDER BY BEGIN, END DESC) END
FROM T_NUM
)
)
WHERE RN = 1
AND CN = 1;

簡(jiǎn)單的說(shuō),內(nèi)層的分析函數(shù)MAX用來(lái)根據(jù)BEGIN從小到大,END從大到小的條件,確定每個(gè)范圍對(duì)應(yīng)的最大的END的值。而外層的兩個(gè)分析函數(shù),COUNT用來(lái)去掉完全重復(fù)的記錄,而ROW_NUMBER用來(lái)獲取范圍最大的記錄(也就是沒(méi)有被其他記錄的范圍所涵蓋)。

改寫(xiě)后,這個(gè)SQL避免對(duì)自關(guān)聯(lián),也就不存在關(guān)聯(lián)條件重復(fù)值過(guò)高的性能隱患了。在模擬環(huán)境中,性能對(duì)比如下:

SQL> SELECT BEGIN, END, ROWID, LENGTH(BEGIN)

2 FROM T_NUM A

3 WHERE NOT EXISTS (

4  SELECT 1

5  FROM T_NUM B

6  WHERE B.BEGIN = A.BEGIN

7  AND B.END >= A.END

8  AND B.ROWID != A.ROWID

9  AND LENGTH(B.BEGIN) = LENGTH(A.BEGIN))

10 ;

 

48344 rows selected.

 

Elapsed: 00:00:57.68

 

Execution Plan

———————————————————-

Plan hash value: 2540751655

 

————————————————————————————

| Id | Operation   | Name | Rows | Bytes |TempSpc| Cost (%CPU)| Time  |

————————————————————————————

| 0 | SELECT STATEMENT |  | 48454 | 1703K|  | 275 (1)| 00:00:04 |

|* 1 | HASH JOIN ANTI |  | 48454 | 1703K| 1424K| 275 (1)| 00:00:04 |

| 2 | TABLE ACCESS FULL| T_NUM | 48454 | 851K|  | 68 (0)| 00:00:01 |

| 3 | TABLE ACCESS FULL| T_NUM | 48454 | 851K|  | 68 (0)| 00:00:01 |

————————————————————————————

 

Predicate Information (identified by operation id):

—————————————————

 

1 – access(LENGTH(TO_CHAR(“B”.”BEGIN”))=LENGTH(TO_CHAR(“A”.”BEGIN”)))

filter(“B”.”BEGIN”=”A”.”BEGIN” AND “B”.”END”>=”A”.”END” AND

“B”.ROWID>”A”.ROWID)

 

 

Statistics

———————————————————-

0 recursive calls

0 db block gets

404 consistent gets

0 physical reads

0 redo size

2315794 bytes sent via SQL*Net to client

35966 bytes received via SQL*Net from client

3224 SQL*Net roundtrips to/from client

0 sorts (memory)

0 sorts (disk)

48344 rows processed

 

SQL> SELECT BEGIN, OLDEND END, LENGTH(BEGIN)

2 FROM (

3  SELECT BEGIN, OLDEND, END, LENGTH(BEGIN), COUNT(*) OVER(PARTITION BY LENGTH(BEGIN), BEGIN, OLDEND) CN,

4    ROW_NUMBER() OVER(PARTITION BY LENGTH(BEGIN), END ORDER BY BEGIN) RN

5  FROM

6  (

7    SELECT BEGIN, END OLDEND, MAX(END) OVER(PARTITION BY LENGTH(BEGIN) ORDER BY BEGIN, END DESC) END

8    FROM T_NUM

9  )

10 )

11 WHERE RN = 1

12 AND CN = 1;

 

48344 rows selected.

 

Elapsed: 00:00:00.72

 

Execution Plan

———————————————————-

Plan hash value: 1546715670

 

——————————————————————————————

| Id | Operation    | Name | Rows | Bytes |TempSpc| Cost (%CPU)| Time  |

——————————————————————————————

| 0 | SELECT STATEMENT   |  | 48454 | 2460K|  | 800 (1)| 00:00:10 |

|* 1 | VIEW     |  | 48454 | 2460K|  | 800 (1)| 00:00:10 |

|* 2 | WINDOW SORT PUSHED RANK|  | 48454 | 1845K| 2480K| 800 (1)| 00:00:10 |

| 3 | WINDOW BUFFER   |  | 48454 | 1845K|  | 800 (1)| 00:00:10 |

| 4 |  VIEW     |  | 48454 | 1845K|  | 311 (1)| 00:00:04 |

| 5 |  WINDOW SORT   |  | 48454 | 662K| 1152K| 311 (1)| 00:00:04 |

| 6 |  TABLE ACCESS FULL | T_NUM | 48454 | 662K|  | 68 (0)| 00:00:01 |

——————————————————————————————

 

Predicate Information (identified by operation id):

—————————————————

 

1 – filter(“RN”=1 AND “CN”=1)

2 – filter(ROW_NUMBER() OVER ( PARTITION BY LENGTH(TO_CHAR(“BEGIN”)),”END”

ORDER BY “BEGIN”)=1)

 

 

Statistics

———————————————————-

0 recursive calls

0 db block gets

202 consistent gets

0 physical reads

0 redo size

1493879 bytes sent via SQL*Net to client

35966 bytes received via SQL*Net from client

3224 SQL*Net roundtrips to/from client

3 sorts (memory)

0 sorts (disk)

48344 rows processed

原SQL運(yùn)行時(shí)間接近1分鐘,而改寫(xiě)后的SQL語(yǔ)句只需要0.72秒,執(zhí)行時(shí)間變?yōu)樵镜?/80,邏輯讀減少一半。

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。

您可能感興趣的文章:
  • MySQL中三種關(guān)聯(lián)查詢方式的簡(jiǎn)單比較
  • mysql中的跨庫(kù)關(guān)聯(lián)查詢方法
  • 淺談mysql中多表不關(guān)聯(lián)查詢的實(shí)現(xiàn)方法
  • MySQL中表子查詢與關(guān)聯(lián)子查詢的基礎(chǔ)學(xué)習(xí)教程
  • 實(shí)例講解Java的MyBatis框架對(duì)MySQL中數(shù)據(jù)的關(guān)聯(lián)查詢
  • MyBatis實(shí)踐之動(dòng)態(tài)SQL及關(guān)聯(lián)查詢
  • mysql關(guān)聯(lián)子查詢的一種優(yōu)化方法分析
  • Yii2實(shí)現(xiàn)跨mysql數(shù)據(jù)庫(kù)關(guān)聯(lián)查詢排序功能代碼
  • mysql一對(duì)多關(guān)聯(lián)查詢分頁(yè)錯(cuò)誤問(wèn)題的解決方法

標(biāo)簽:荊門(mén) 柳州 淮安 景德鎮(zhèn) 江蘇 威海 那曲 瀘州

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《如何利用分析函數(shù)改寫(xiě)范圍判斷自關(guān)聯(lián)查詢?cè)斀狻?,本文關(guān)鍵詞  如何,利用,分析,函數(shù),改寫(xiě),;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《如何利用分析函數(shù)改寫(xiě)范圍判斷自關(guān)聯(lián)查詢?cè)斀狻废嚓P(guān)的同類信息!
  • 本頁(yè)收集關(guān)于如何利用分析函數(shù)改寫(xiě)范圍判斷自關(guān)聯(lián)查詢?cè)斀獾南嚓P(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    虹口区| 沿河| 安达市| 黄浦区| 义乌市| 杭锦旗| 普陀区| 洛浦县| 开平市| 宁国市| 察雅县| 巫山县| 绥棱县| 临潭县| 文成县| 灵川县| 施秉县| 壶关县| 神农架林区| 阿城市| 上饶市| 兖州市| 黎平县| 建水县| 安乡县| 美姑县| 沧源| 深圳市| 宜黄县| 武乡县| 山阴县| 苍山县| 大庆市| 米林县| 平阴县| 甘南县| 嘉义县| 平武县| 海兴县| 汾阳市| 固阳县|