如何利用分析函數(shù)改寫(xiě)范圍判斷自關(guān)聯(lián)查詢(xún)?cè)斀?/h1>
更新時(shí)間:2018年10月13日 11:28:58 作者:yangtingkun
這篇文章主要給大家介紹了關(guān)于如何利用分析函數(shù)改寫(xiě)范圍判斷自關(guān)聯(lián)查詢(xún)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用sql具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
前言
最近碰到一個(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文本類(lèi)似如下:
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)的選擇度非常查,而其他的條件都是不等查詢(xún),選擇度也不會(huì)好,即使建立索引,強(qiáng)制執(zhí)行選擇索引,效率也不會(huì)好。
那么如果想要繼續(xù)優(yōu)化這個(gè)SQL,就只剩下一個(gè)辦法,那就是SQL的改寫(xiě)。對(duì)于自關(guān)聯(lián)查詢(xú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)查詢(xún)方式的簡(jiǎn)單比較
- mysql中的跨庫(kù)關(guān)聯(lián)查詢(xún)方法
- 淺談mysql中多表不關(guān)聯(lián)查詢(xún)的實(shí)現(xiàn)方法
- MySQL中表子查詢(xún)與關(guān)聯(lián)子查詢(xún)的基礎(chǔ)學(xué)習(xí)教程
- 實(shí)例講解Java的MyBatis框架對(duì)MySQL中數(shù)據(jù)的關(guān)聯(lián)查詢(xún)
- MyBatis實(shí)踐之動(dòng)態(tài)SQL及關(guān)聯(lián)查詢(xún)
- mysql關(guān)聯(lián)子查詢(xún)的一種優(yōu)化方法分析
- Yii2實(shí)現(xiàn)跨mysql數(shù)據(jù)庫(kù)關(guān)聯(lián)查詢(xún)排序功能代碼
- mysql一對(duì)多關(guān)聯(lián)查詢(xún)分頁(yè)錯(cuò)誤問(wèn)題的解決方法
相關(guān)文章
-
使用SQL Mail收發(fā)和自動(dòng)處理郵件中的擴(kuò)展存儲(chǔ)過(guò)程簡(jiǎn)介
使用SQL Mail收發(fā)和自動(dòng)處理郵件中的擴(kuò)展存儲(chǔ)過(guò)程簡(jiǎn)介... 2006-12-12
-
Navicat Premium 16最新永久激活教程(NavicatCracker)
最新版的Navicat Premium 16 已經(jīng)發(fā)布,今天小編給大家分享Navicat Premium 16最新永久激活教程(NavicatCracker),感興趣的朋友跟隨小編一起看看吧 2023-06-06
-
DBeaver一款替代Navicat的數(shù)據(jù)庫(kù)可視化工具
這篇文章主要介紹了DBeaver一款替代Navicat的數(shù)據(jù)庫(kù)可視化工具,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧 2020-11-11
-
sql學(xué)習(xí)之CASE WHEN THEN ELSE END的用法
這篇文章主要介紹了sql學(xué)習(xí)之CASE WHEN THEN ELSE END的用法,需要的朋友可以參考下 2014-06-06
-
程序員應(yīng)該知道的數(shù)據(jù)庫(kù)設(shè)計(jì)的兩個(gè)誤區(qū)
在幾乎所有的企業(yè)級(jí)應(yīng)用程序中,包括各種MIS、ERP、CRM等等,都會(huì)使用數(shù)據(jù)庫(kù),這樣的好處是顯而易見(jiàn)的,很容易地實(shí)現(xiàn)了數(shù)據(jù)層和業(yè)務(wù)邏輯層的分離,而且對(duì)于性能的優(yōu)化也在一定程度上提供了便利。 2010-07-07
-
如何判斷a、b、c三個(gè)字段同時(shí)為0則不顯示這條數(shù)據(jù)
有時(shí)候我們需要判斷當(dāng)a、b、c三個(gè)字段同時(shí)為0則不顯示,下面這個(gè)方法不錯(cuò),需要的朋友可以參考下 2013-08-08
前言
最近碰到一個(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文本類(lèi)似如下:
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)的選擇度非常查,而其他的條件都是不等查詢(xún),選擇度也不會(huì)好,即使建立索引,強(qiáng)制執(zhí)行選擇索引,效率也不會(huì)好。
那么如果想要繼續(xù)優(yōu)化這個(gè)SQL,就只剩下一個(gè)辦法,那就是SQL的改寫(xiě)。對(duì)于自關(guān)聯(lián)查詢(xú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)查詢(xún)方式的簡(jiǎn)單比較
- mysql中的跨庫(kù)關(guān)聯(lián)查詢(xún)方法
- 淺談mysql中多表不關(guān)聯(lián)查詢(xún)的實(shí)現(xiàn)方法
- MySQL中表子查詢(xún)與關(guān)聯(lián)子查詢(xún)的基礎(chǔ)學(xué)習(xí)教程
- 實(shí)例講解Java的MyBatis框架對(duì)MySQL中數(shù)據(jù)的關(guān)聯(lián)查詢(xún)
- MyBatis實(shí)踐之動(dòng)態(tài)SQL及關(guān)聯(lián)查詢(xún)
- mysql關(guān)聯(lián)子查詢(xún)的一種優(yōu)化方法分析
- Yii2實(shí)現(xiàn)跨mysql數(shù)據(jù)庫(kù)關(guān)聯(lián)查詢(xún)排序功能代碼
- mysql一對(duì)多關(guān)聯(lián)查詢(xún)分頁(yè)錯(cuò)誤問(wèn)題的解決方法
相關(guān)文章
使用SQL Mail收發(fā)和自動(dòng)處理郵件中的擴(kuò)展存儲(chǔ)過(guò)程簡(jiǎn)介
使用SQL Mail收發(fā)和自動(dòng)處理郵件中的擴(kuò)展存儲(chǔ)過(guò)程簡(jiǎn)介...2006-12-12Navicat Premium 16最新永久激活教程(NavicatCracker)
最新版的Navicat Premium 16 已經(jīng)發(fā)布,今天小編給大家分享Navicat Premium 16最新永久激活教程(NavicatCracker),感興趣的朋友跟隨小編一起看看吧2023-06-06DBeaver一款替代Navicat的數(shù)據(jù)庫(kù)可視化工具
這篇文章主要介紹了DBeaver一款替代Navicat的數(shù)據(jù)庫(kù)可視化工具,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11sql學(xué)習(xí)之CASE WHEN THEN ELSE END的用法
這篇文章主要介紹了sql學(xué)習(xí)之CASE WHEN THEN ELSE END的用法,需要的朋友可以參考下2014-06-06程序員應(yīng)該知道的數(shù)據(jù)庫(kù)設(shè)計(jì)的兩個(gè)誤區(qū)
在幾乎所有的企業(yè)級(jí)應(yīng)用程序中,包括各種MIS、ERP、CRM等等,都會(huì)使用數(shù)據(jù)庫(kù),這樣的好處是顯而易見(jiàn)的,很容易地實(shí)現(xiàn)了數(shù)據(jù)層和業(yè)務(wù)邏輯層的分離,而且對(duì)于性能的優(yōu)化也在一定程度上提供了便利。2010-07-07如何判斷a、b、c三個(gè)字段同時(shí)為0則不顯示這條數(shù)據(jù)
有時(shí)候我們需要判斷當(dāng)a、b、c三個(gè)字段同時(shí)為0則不顯示,下面這個(gè)方法不錯(cuò),需要的朋友可以參考下2013-08-08