昨天刚写了半连接改写系列,今天就遇到了此类型SQL:

优化前

耗时:28s

返回:0

SELECT D.DAILYAUDITNO, D.TRANSTOACC
FROM PB_DOIC.MM_DAILYREPORT_TD D
WHERE D.REPORTSTATUS = '4'
AND D.TRANSTOACC IN ('00', 'B1')
AND EXISTS (SELECT 1
FROM PB_DOIC.MM_DAILYREPORT_DETAIL_TD
WHERE DAILYAUDITNO = D.DAILYAUDITNO
AND BUSINESSONE || BUSINESSTWO NOT IN
('604988', '605988', '606988', '607988')
AND BUSINESSTWO != '991')
AND NOT EXISTS
(SELECT 1
FROM PB_DOIC.MM_DAILYREPORT_DETAIL_TD E
WHERE E.DAILYAUDITNO = D.DAILYAUDITNO
AND E.BUSINESSONE || E.BUSINESSTWO NOT IN
('604988', '605988', '606988', '607988')
AND E.BUSINESSTWO != '991'
AND NOT EXISTS
(SELECT 1
FROM PB_DOIC.MM_VOUCHERMODULE_TC V
WHERE V.BUSINESSTYPE = E.BUSINESSONE || E.BUSINESSTWO
AND V.MODULETYPE = '1'
AND V.MODULESTATUS = 0))
FOR UPDATE OF D.TRANSTOACC NOWAIT Execution Plan
----------------------------------------------------------
Plan hash value: 915590932 --------------------------------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
--------------------------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 10 | 1160 | 20991 (2)| 00:04:12 |
| 1 | FOR UPDATE | | | | | |
| 2 | NESTED LOOPS | | 1 | 116 | 20945 (2)| 00:04:12 |
| 3 | SORT UNIQUE | | 29 | 1943 | 20928 (2)| 00:04:12 |
|* 4 | INDEX FAST FULL SCAN | IDX_MM_DAILYREPORT_DETAIL_TD | 29 | 1943 | 20928 (2)| 00:04:12 |
|* 5 | TABLE ACCESS BY INDEX ROWID| MM_DAILYREPORT_TD | 1 | 49 | 2 (0)| 00:00:01 |
|* 6 | INDEX UNIQUE SCAN | PK_MM_DAILYREPORT_TD | 1 | | 1 (0)| 00:00:01 |
|* 7 | INDEX RANGE SCAN | IDX_MM_DAILYREPORT_DETAIL_TD | 1 | 67 | 7 (0)| 00:00:01 |
|* 8 | INDEX RANGE SCAN | PK_MM_VOUCHERMODULE_TC | 1 | 15 | 2 (0)| 00:00:01 |
-------------------------------------------------------------------------------------------------------------- Predicate Information (identified by operation id):
--------------------------------------------------- 4 - filter("BUSINESSONE"||"BUSINESSTWO"<>'604988' AND "BUSINESSONE"||"BUSINESSTWO"<>'605988' AND
"BUSINESSONE"||"BUSINESSTWO"<>'606988' AND "BUSINESSONE"||"BUSINESSTWO"<>'607988' AND
"BUSINESSTWO"<>'991')
5 - filter(("D"."TRANSTOACC"='00' OR "D"."TRANSTOACC"='B1') AND "D"."REPORTSTATUS"='4')
6 - access("DAILYAUDITNO"="D"."DAILYAUDITNO")
filter( NOT EXISTS (SELECT /*+ */ 0 FROM "PB_DOIC"."MM_DAILYREPORT_DETAIL_TD" "SYS_ALIAS_2"
WHERE "E"."DAILYAUDITNO"=:B1 AND "E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'604988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'605988' AND "E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'606988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'607988' AND "E"."BUSINESSTWO"<>'991' AND NOT EXISTS (SELECT
/*+ */ 0 FROM "PB_DOIC"."MM_VOUCHERMODULE_TC" "V" WHERE "V"."BUSINESSTYPE"=:B2||:B3 AND
"V"."MODULETYPE"='1' AND "V"."MODULESTATUS"=0)))
7 - access("E"."DAILYAUDITNO"=:B1)
filter("E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'604988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'605988' AND "E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'606988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'607988' AND "E"."BUSINESSTWO"<>'991' AND NOT EXISTS (SELECT
/*+ */ 0 FROM "PB_DOIC"."MM_VOUCHERMODULE_TC" "V" WHERE "V"."BUSINESSTYPE"=:B1||:B2 AND
"V"."MODULETYPE"='1' AND "V"."MODULESTATUS"=0))
8 - access("V"."MODULETYPE"='1' AND "V"."BUSINESSTYPE"=:B1||:B2 AND "V"."MODULESTATUS"=0)
filter("V"."MODULESTATUS"=0) Statistics
----------------------------------------------------------
1 recursive calls
0 db block gets
3809828 consistent gets
347 physical reads
0 redo size
397 bytes sent via SQL*Net to client
481 bytes received via SQL*Net from client
1 SQL*Net roundtrips to/from client
1 sorts (memory)
0 sorts (disk)
0 rows processed

分析

查看执行计划,发现ID=3处为 SORT UNIQUE ,而SQL语句并没有DISTINCT关键字,难道SQL进行自动改写了?

为了确认自己想法,我进行以下手动改写,把半连接手动改写成内连接:

改写之前,先确认表与表之间的关系:

select count(*),count(distinct DAILYAUDITNO) from MM_DAILYREPORT_DETAIL_TD;

COUNT(*)    COUNT(DISTINCTDAILYAUDITNO)
-------- -----------------------
5111081 441898 select count(*),count(distinct DAILYAUDITNO) from MM_DAILYREPORT_TD; COUNT(*) COUNT(DISTINCTDAILYAUDITNO)
-------- -----------------------
441940 441940 --表MM_DAILYREPORT_TD与表MM_DAILYREPORT_DETAIL_TD是1:N的关系,所以改写时候,需要先去重,再连接。 SELECT D.DAILYAUDITNO, D.TRANSTOACC
FROM PB_DOIC.MM_DAILYREPORT_TD D
INNER JOIN (SELECT DISTINCT DAILYAUDITNO
FROM PB_DOIC.MM_DAILYREPORT_DETAIL_TD
WHERE BUSINESSONE || BUSINESSTWO NOT IN
('604988', '605988', '606988', '607988')
AND BUSINESSTWO != '991') TD
ON D.DAILYAUDITNO = TD.DAILYAUDITNO
WHERE D.REPORTSTATUS = '4'
AND D.TRANSTOACC IN ('00', 'B1')
AND NOT EXISTS
(SELECT 1
FROM PB_DOIC.MM_DAILYREPORT_DETAIL_TD E
WHERE E.DAILYAUDITNO = D.DAILYAUDITNO
AND E.BUSINESSONE || E.BUSINESSTWO NOT IN
('604988', '605988', '606988', '607988')
AND E.BUSINESSTWO != '991'
AND NOT EXISTS
(SELECT 1
FROM PB_DOIC.MM_VOUCHERMODULE_TC V
WHERE V.BUSINESSTYPE = E.BUSINESSONE || E.BUSINESSTWO
AND V.MODULETYPE = '1'
AND V.MODULESTATUS = 0))
FOR UPDATE OF D.TRANSTOACC NOWAIT; Execution Plan
----------------------------------------------------------
Plan hash value: 1310135718 --------------------------------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
--------------------------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 25 | 2250 | 21076 (2)| 00:04:13 |
| 1 | FOR UPDATE | | | | | |
| 2 | NESTED LOOPS | | 1 | 90 | 20959 (2)| 00:04:12 |
| 3 | VIEW | | 29 | 1189 | 20929 (2)| 00:04:12 |
| 4 | SORT UNIQUE | | 29 | 1943 | 20929 (2)| 00:04:12 |
|* 5 | INDEX FAST FULL SCAN | IDX_MM_DAILYREPORT_DETAIL_TD | 29 | 1943 | 20928 (2)| 00:04:12 |
|* 6 | TABLE ACCESS BY INDEX ROWID| MM_DAILYREPORT_TD | 1 | 49 | 2 (0)| 00:00:01 |
|* 7 | INDEX UNIQUE SCAN | PK_MM_DAILYREPORT_TD | 1 | | 1 (0)| 00:00:01 |
|* 8 | INDEX RANGE SCAN | IDX_MM_DAILYREPORT_DETAIL_TD | 1 | 67 | 7 (0)| 00:00:01 |
|* 9 | INDEX RANGE SCAN | PK_MM_VOUCHERMODULE_TC | 1 | 15 | 2 (0)| 00:00:01 |
-------------------------------------------------------------------------------------------------------------- Predicate Information (identified by operation id):
--------------------------------------------------- 5 - filter("BUSINESSONE"||"BUSINESSTWO"<>'604988' AND "BUSINESSONE"||"BUSINESSTWO"<>'605988' AND
"BUSINESSONE"||"BUSINESSTWO"<>'606988' AND "BUSINESSONE"||"BUSINESSTWO"<>'607988' AND
"BUSINESSTWO"<>'991')
6 - filter(("D"."TRANSTOACC"='00' OR "D"."TRANSTOACC"='B1') AND "D"."REPORTSTATUS"='4')
7 - access("D"."DAILYAUDITNO"="TD"."DAILYAUDITNO")
filter( NOT EXISTS (SELECT /*+ */ 0 FROM "PB_DOIC"."MM_DAILYREPORT_DETAIL_TD" "SYS_ALIAS_2"
WHERE "E"."DAILYAUDITNO"=:B1 AND "E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'604988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'605988' AND "E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'606988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'607988' AND "E"."BUSINESSTWO"<>'991' AND NOT EXISTS (SELECT
/*+ */ 0 FROM "PB_DOIC"."MM_VOUCHERMODULE_TC" "V" WHERE "V"."BUSINESSTYPE"=:B2||:B3 AND
"V"."MODULETYPE"='1' AND "V"."MODULESTATUS"=0)))
8 - access("E"."DAILYAUDITNO"=:B1)
filter("E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'604988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'605988' AND "E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'606988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'607988' AND "E"."BUSINESSTWO"<>'991' AND NOT EXISTS (SELECT
/*+ */ 0 FROM "PB_DOIC"."MM_VOUCHERMODULE_TC" "V" WHERE "V"."BUSINESSTYPE"=:B1||:B2 AND
"V"."MODULETYPE"='1' AND "V"."MODULESTATUS"=0))
9 - access("V"."MODULETYPE"='1' AND "V"."BUSINESSTYPE"=:B1||:B2 AND "V"."MODULESTATUS"=0)
filter("V"."MODULESTATUS"=0) Statistics
----------------------------------------------------------
1 recursive calls
0 db block gets
3809828 consistent gets
431 physical reads
0 redo size
397 bytes sent via SQL*Net to client
481 bytes received via SQL*Net from client
1 SQL*Net roundtrips to/from client
1 sorts (memory)
0 sorts (disk)
0 rows processed

可以看出改写后的执行计划于改写之前执行计划基本一致,所以可以确定SQL是进行自动改写内连接而导致的性能问题。

优化方法1:

耗时:0.4s

返回:0

--在半连接里加上ROWNUM>0,让SQL不进行展开。
SELECT D.DAILYAUDITNO, D.TRANSTOACC
FROM PB_DOIC.MM_DAILYREPORT_TD D
WHERE D.REPORTSTATUS = '4'
AND D.TRANSTOACC IN ('00', 'B1')
AND EXISTS (SELECT 1
FROM PB_DOIC.MM_DAILYREPORT_DETAIL_TD
WHERE DAILYAUDITNO = D.DAILYAUDITNO
AND BUSINESSONE || BUSINESSTWO NOT IN
('604988', '605988', '606988', '607988')
AND BUSINESSTWO != '991' AND ROWNUM>0)
AND NOT EXISTS
(SELECT 1
FROM PB_DOIC.MM_DAILYREPORT_DETAIL_TD E
WHERE E.DAILYAUDITNO = D.DAILYAUDITNO
AND E.BUSINESSONE || E.BUSINESSTWO NOT IN
('604988', '605988', '606988', '607988')
AND E.BUSINESSTWO != '991'
AND NOT EXISTS
(SELECT 1
FROM PB_DOIC.MM_VOUCHERMODULE_TC V
WHERE V.BUSINESSTYPE = E.BUSINESSONE || E.BUSINESSTWO
AND V.MODULETYPE = '1'
AND V.MODULESTATUS = 0))
FOR UPDATE OF D.TRANSTOACC NOWAIT; Execution Plan
----------------------------------------------------------
Plan hash value: 4088075167 ------------------------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
------------------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 49 | 336K (1)| 01:07:15 |
| 1 | FOR UPDATE | | | | | |
|* 2 | FILTER | | | | | |
|* 3 | TABLE ACCESS FULL | MM_DAILYREPORT_TD | 22372 | 1070K| 2632 (2)| 00:00:32 |
| 4 | COUNT | | | | | |
|* 5 | FILTER | | | | | |
|* 6 | INDEX RANGE SCAN | IDX_MM_DAILYREPORT_DETAIL_TD | 1 | 67 | 7 (0)| 00:00:01 |
|* 7 | INDEX RANGE SCAN | IDX_MM_DAILYREPORT_DETAIL_TD | 1 | 67 | 7 (0)| 00:00:01 |
|* 8 | INDEX RANGE SCAN| PK_MM_VOUCHERMODULE_TC | 1 | 15 | 2 (0)| 00:00:01 |
------------------------------------------------------------------------------------------------------ Predicate Information (identified by operation id):
--------------------------------------------------- 2 - filter( EXISTS (SELECT /*+ */ 0 FROM "PB_DOIC"."MM_DAILYREPORT_DETAIL_TD"
"MM_DAILYREPORT_DETAIL_TD" WHERE ROWNUM>0 AND "DAILYAUDITNO"=:B1 AND
"BUSINESSONE"||"BUSINESSTWO"<>'604988' AND "BUSINESSONE"||"BUSINESSTWO"<>'605988' AND
"BUSINESSONE"||"BUSINESSTWO"<>'606988' AND "BUSINESSONE"||"BUSINESSTWO"<>'607988' AND
"BUSINESSTWO"<>'991') AND NOT EXISTS (SELECT /*+ */ 0 FROM
"PB_DOIC"."MM_DAILYREPORT_DETAIL_TD" "SYS_ALIAS_2" WHERE "E"."DAILYAUDITNO"=:B2 AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'604988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'605988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'606988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'607988' AND "E"."BUSINESSTWO"<>'991' AND NOT EXISTS
(SELECT /*+ */ 0 FROM "PB_DOIC"."MM_VOUCHERMODULE_TC" "V" WHERE "V"."BUSINESSTYPE"=:B3||:B4
AND "V"."MODULETYPE"='1' AND "V"."MODULESTATUS"=0)))
3 - filter(("D"."TRANSTOACC"='00' OR "D"."TRANSTOACC"='B1') AND "D"."REPORTSTATUS"='4')
5 - filter(ROWNUM>0)
6 - access("DAILYAUDITNO"=:B1)
filter("BUSINESSONE"||"BUSINESSTWO"<>'604988' AND
"BUSINESSONE"||"BUSINESSTWO"<>'605988' AND "BUSINESSONE"||"BUSINESSTWO"<>'606988' AND
"BUSINESSONE"||"BUSINESSTWO"<>'607988' AND "BUSINESSTWO"<>'991')
7 - access("E"."DAILYAUDITNO"=:B1)
filter("E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'604988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'605988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'606988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'607988' AND "E"."BUSINESSTWO"<>'991' AND NOT EXISTS
(SELECT /*+ */ 0 FROM "PB_DOIC"."MM_VOUCHERMODULE_TC" "V" WHERE "V"."BUSINESSTYPE"=:B1||:B2
AND "V"."MODULETYPE"='1' AND "V"."MODULESTATUS"=0))
8 - access("V"."MODULETYPE"='1' AND "V"."BUSINESSTYPE"=:B1||:B2 AND "V"."MODULESTATUS"=0)
filter("V"."MODULESTATUS"=0) Statistics
----------------------------------------------------------
0 recursive calls
0 db block gets
105048 consistent gets
0 physical reads
0 redo size
397 bytes sent via SQL*Net to client
481 bytes received via SQL*Net from client
1 SQL*Net roundtrips to/from client
0 sorts (memory)
0 sorts (disk)
0 rows processed

优化方法2:

耗时:0.4s

返回:0

--添加hint: /*+ no_unnest */ 目的也是让半连接的sql不展开。与上面改写效果一致。

SELECT D.DAILYAUDITNO, D.TRANSTOACC
FROM PB_DOIC.MM_DAILYREPORT_TD D
WHERE D.REPORTSTATUS = '4'
AND D.TRANSTOACC IN ('00', 'B1')
AND EXISTS (SELECT /*+NO_UNNEST*/1
FROM PB_DOIC.MM_DAILYREPORT_DETAIL_TD
WHERE DAILYAUDITNO = D.DAILYAUDITNO
AND BUSINESSONE || BUSINESSTWO NOT IN
('604988', '605988', '606988', '607988')
AND BUSINESSTWO != '991')
AND NOT EXISTS
(SELECT 1
FROM PB_DOIC.MM_DAILYREPORT_DETAIL_TD E
WHERE E.DAILYAUDITNO = D.DAILYAUDITNO
AND E.BUSINESSONE || E.BUSINESSTWO NOT IN
('604988', '605988', '606988', '607988')
AND E.BUSINESSTWO != '991'
AND NOT EXISTS
(SELECT 1
FROM PB_DOIC.MM_VOUCHERMODULE_TC V
WHERE V.BUSINESSTYPE = E.BUSINESSONE || E.BUSINESSTWO
AND V.MODULETYPE = '1'
AND V.MODULESTATUS = 0))
FOR UPDATE OF D.TRANSTOACC NOWAIT Execution Plan
----------------------------------------------------------
Plan hash value: 991259826 -----------------------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
-----------------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 49 | 336K (1)| 01:07:15 |
| 1 | FOR UPDATE | | | | | |
|* 2 | FILTER | | | | | |
|* 3 | TABLE ACCESS FULL | MM_DAILYREPORT_TD | 22372 | 1070K| 2632 (2)| 00:00:32 |
|* 4 | INDEX RANGE SCAN | IDX_MM_DAILYREPORT_DETAIL_TD | 1 | 67 | 7 (0)| 00:00:01 |
|* 5 | INDEX RANGE SCAN | IDX_MM_DAILYREPORT_DETAIL_TD | 1 | 67 | 7 (0)| 00:00:01 |
|* 6 | INDEX RANGE SCAN| PK_MM_VOUCHERMODULE_TC | 1 | 15 | 2 (0)| 00:00:01 |
----------------------------------------------------------------------------------------------------- Predicate Information (identified by operation id):
--------------------------------------------------- 2 - filter( EXISTS (SELECT /*+ NO_UNNEST */ 0 FROM "PB_DOIC"."MM_DAILYREPORT_DETAIL_TD"
"MM_DAILYREPORT_DETAIL_TD" WHERE "DAILYAUDITNO"=:B1 AND
"BUSINESSONE"||"BUSINESSTWO"<>'604988' AND "BUSINESSONE"||"BUSINESSTWO"<>'605988' AND
"BUSINESSONE"||"BUSINESSTWO"<>'606988' AND "BUSINESSONE"||"BUSINESSTWO"<>'607988' AND
"BUSINESSTWO"<>'991') AND NOT EXISTS (SELECT /*+ */ 0 FROM
"PB_DOIC"."MM_DAILYREPORT_DETAIL_TD" "SYS_ALIAS_2" WHERE "E"."DAILYAUDITNO"=:B2 AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'604988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'605988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'606988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'607988' AND "E"."BUSINESSTWO"<>'991' AND NOT EXISTS
(SELECT /*+ */ 0 FROM "PB_DOIC"."MM_VOUCHERMODULE_TC" "V" WHERE "V"."BUSINESSTYPE"=:B3||:B4
AND "V"."MODULETYPE"='1' AND "V"."MODULESTATUS"=0)))
3 - filter(("D"."TRANSTOACC"='00' OR "D"."TRANSTOACC"='B1') AND "D"."REPORTSTATUS"='4')
4 - access("DAILYAUDITNO"=:B1)
filter("BUSINESSONE"||"BUSINESSTWO"<>'604988' AND
"BUSINESSONE"||"BUSINESSTWO"<>'605988' AND "BUSINESSONE"||"BUSINESSTWO"<>'606988' AND
"BUSINESSONE"||"BUSINESSTWO"<>'607988' AND "BUSINESSTWO"<>'991')
5 - access("E"."DAILYAUDITNO"=:B1)
filter("E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'604988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'605988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'606988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'607988' AND "E"."BUSINESSTWO"<>'991' AND NOT EXISTS
(SELECT /*+ */ 0 FROM "PB_DOIC"."MM_VOUCHERMODULE_TC" "V" WHERE "V"."BUSINESSTYPE"=:B1||:B2
AND "V"."MODULETYPE"='1' AND "V"."MODULESTATUS"=0))
6 - access("V"."MODULETYPE"='1' AND "V"."BUSINESSTYPE"=:B1||:B2 AND "V"."MODULESTATUS"=0)
filter("V"."MODULESTATUS"=0) Statistics
----------------------------------------------------------
1 recursive calls
0 db block gets
105048 consistent gets
0 physical reads
0 redo size
397 bytes sent via SQL*Net to client
481 bytes received via SQL*Net from client
1 SQL*Net roundtrips to/from client
0 sorts (memory)
0 sorts (disk)
0 rows processed

优化方法3:

耗时:0.4s

返回:0

添加hint:/*+ nl_sj */,目的让SQL走semi join

SELECT D.DAILYAUDITNO, D.TRANSTOACC
FROM PB_DOIC.MM_DAILYREPORT_TD D
WHERE D.REPORTSTATUS = '4'
AND D.TRANSTOACC IN ('00', 'B1')
AND EXISTS (SELECT /*+nl_sj*/ 1
FROM PB_DOIC.MM_DAILYREPORT_DETAIL_TD
WHERE DAILYAUDITNO = D.DAILYAUDITNO
AND BUSINESSONE || BUSINESSTWO NOT IN
('604988', '605988', '606988', '607988')
AND BUSINESSTWO != '991')
AND NOT EXISTS
(SELECT 1
FROM PB_DOIC.MM_DAILYREPORT_DETAIL_TD E
WHERE E.DAILYAUDITNO = D.DAILYAUDITNO
AND E.BUSINESSONE || E.BUSINESSTWO NOT IN
('604988', '605988', '606988', '607988')
AND E.BUSINESSTWO != '991'
AND NOT EXISTS
(SELECT 1
FROM PB_DOIC.MM_VOUCHERMODULE_TC V
WHERE V.BUSINESSTYPE = E.BUSINESSONE || E.BUSINESSTWO
AND V.MODULETYPE = '1'
AND V.MODULESTATUS = 0))
FOR UPDATE OF D.TRANSTOACC NOWAIT Execution Plan
----------------------------------------------------------
Plan hash value: 2686500646 -----------------------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
-----------------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 34 | 3944 | 137K (1)| 00:27:29 |
| 1 | FOR UPDATE | | | | | |
|* 2 | FILTER | | | | | |
| 3 | NESTED LOOPS SEMI | | 34 | 3944 | 137K (1)| 00:27:27 |
|* 4 | TABLE ACCESS FULL| MM_DAILYREPORT_TD | 22372 | 1070K| 2632 (2)| 00:00:32 |
|* 5 | INDEX RANGE SCAN | IDX_MM_DAILYREPORT_DETAIL_TD | 1 | 67 | 6 (0)| 00:00:01 |
|* 6 | INDEX RANGE SCAN | IDX_MM_DAILYREPORT_DETAIL_TD | 1 | 67 | 7 (0)| 00:00:01 |
|* 7 | INDEX RANGE SCAN | PK_MM_VOUCHERMODULE_TC | 1 | 15 | 2 (0)| 00:00:01 |
----------------------------------------------------------------------------------------------------- Predicate Information (identified by operation id):
--------------------------------------------------- 2 - filter( NOT EXISTS (SELECT /*+ */ 0 FROM "PB_DOIC"."MM_DAILYREPORT_DETAIL_TD"
"SYS_ALIAS_2" WHERE "E"."DAILYAUDITNO"=:B1 AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'604988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'605988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'606988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'607988' AND "E"."BUSINESSTWO"<>'991' AND NOT EXISTS
(SELECT /*+ */ 0 FROM "PB_DOIC"."MM_VOUCHERMODULE_TC" "V" WHERE "V"."BUSINESSTYPE"=:B2||:B3
AND "V"."MODULETYPE"='1' AND "V"."MODULESTATUS"=0)))
4 - filter(("D"."TRANSTOACC"='00' OR "D"."TRANSTOACC"='B1') AND "D"."REPORTSTATUS"='4')
5 - access("DAILYAUDITNO"="D"."DAILYAUDITNO")
filter("BUSINESSONE"||"BUSINESSTWO"<>'604988' AND
"BUSINESSONE"||"BUSINESSTWO"<>'605988' AND "BUSINESSONE"||"BUSINESSTWO"<>'606988' AND
"BUSINESSONE"||"BUSINESSTWO"<>'607988' AND "BUSINESSTWO"<>'991')
6 - access("E"."DAILYAUDITNO"=:B1)
filter("E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'604988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'605988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'606988' AND
"E"."BUSINESSONE"||"E"."BUSINESSTWO"<>'607988' AND "E"."BUSINESSTWO"<>'991' AND NOT EXISTS
(SELECT /*+ */ 0 FROM "PB_DOIC"."MM_VOUCHERMODULE_TC" "V" WHERE "V"."BUSINESSTYPE"=:B1||:B2
AND "V"."MODULETYPE"='1' AND "V"."MODULESTATUS"=0))
7 - access("V"."MODULETYPE"='1' AND "V"."BUSINESSTYPE"=:B1||:B2 AND "V"."MODULESTATUS"=0)
filter("V"."MODULESTATUS"=0) Statistics
----------------------------------------------------------
2114 recursive calls
0 db block gets
83115 consistent gets
1341 physical reads
0 redo size
397 bytes sent via SQL*Net to client
481 bytes received via SQL*Net from client
1 SQL*Net roundtrips to/from client
77 sorts (memory)
0 sorts (disk)
0 rows processed

处理半连接SQL自动改写内连接SQL一例的更多相关文章

  1. Sql server中内连接语句

    数据库中学生表和课程表如下: 内连接sql语句: select a.studentName,a.studentAge,b.courseName from student a inner join co ...

  2. SQL Server中内连接和外连接的区别

    SQL Server中内连接和外连接的区别 假设一个数据库中有两张表,一张是学生表StudentInfo,一张是班级表ClassInfo,两张表之间用ClassId字段进行关联. 如果用内连接,正常的 ...

  3. 你真的会玩SQL吗?内连接、外连接

    原文:你真的会玩SQL吗?内连接.外连接 大多数人一般写多表查询会这样写select * from tbA ,tbB  没有用到JOIN关键字,太Low了,官网标准建议是用JOIN明确表间的关系,下面 ...

  4. SQL中的内连接与外连接

    关于关系代数连接运算的介绍请查看下面链接 http://www.cnblogs.com/xidongyu/articles/5980407.html 连接运算格式 链接运算由两部分构成:连接类型和连接 ...

  5. Sql中的内连接,左连接以及右连接区别

    转自:http://pangaoyuan.javaeye.com/blog/713177 有两个表A和表B. 表A结构如下: Aid:int:标识种子,主键,自增ID Aname:varchar 数据 ...

  6. 优化子查询sql语句为内连接

    背景: 希望提高查询的效率,从sql语句中频繁出现的子查询入手. 数据表如下:Student表中的CityCode对应于City表中的Code. Student表:                   ...

  7. SQL 交叉连接与内连接

    交叉连接 ,没有任何限制方式的连接. 叫做交叉连接. 碰到一种SQL 的写法. select * from  t1,t2 .     这其实是交叉连接 .   t1  是三条 ,  t2 是两条.  ...

  8. SQL中的内连接外连接和交叉连接是什么意思?

    内连接又分为等值连接.自然连接和不等连接三种. 外连接分为左外连接(LEFT OUTER JOIN或LEFT JOIN).右外连接(RIGHT OUTER JOIN或RIGHT JOIN)和全外连接( ...

  9. 清晰讲解SQL语句中的内连接,通用于Mysql和Oracle,全是干货哦

    本文章目的:力求清晰明了讲解SQL语句的内连接的各种应用,没有深奥的理解! 前奏:这篇文章和下篇文章会将内连接和外连接讲解清楚SQL语句的多表查询常用的有以下几种:两表联合查询(1)内连接(2)外连接 ...

随机推荐

  1. Codeforces Round #259 (Div. 1)A(公式)

    传送门 题意 给出m个面的骰子扔n次,取最大值,求期望 分析 暴力算会有重复,而且复杂度不对. 考虑m个面扔n次得到m的概率,发现只要减去(m-1)个面扔n次得到m-1的概率即可,给出example说 ...

  2. bzoj 2986: Non-Squarefree Numbers【容斥+莫比乌斯函数】

    看到\( 10^10 \)的范围首先想到二分,然后把问题转化为判断\( [1,n] \)内有多少个是某个质数的平方和的数. 所以应该是加上是一个质数的平方的个数减去是两个质数的平方的个数加上是三个质数 ...

  3. jQuery 动作效果

    隐藏和显示 jQuery hide() 和 show() 通过 jQuery,您可以使用 hide() 和 show() 方法来隐藏和显示 HTML 元素: jQuery toggle() 通过 jQ ...

  4. [hdu4089] Activation【概率dp 数学期望】

    传送门:http://acm.hdu.edu.cn/showproblem.php?pid=4089 本来可以一遍过的,结果mle了一发...注意要用滚动数组. 令f(i, j)表示队列剩余i个人,这 ...

  5. Ubuntu下如何用命令运行deb安装包

    转载自 WindTaiL的博客 如果ubuntu要安装新软件,已有deb安装包(例如:iptux.deb),但是无法登录到桌面环境.那该怎么安装?答案是:使用dpkg命令. dpkg命令常用格式如下: ...

  6. Service官方教程(7)Bound Service示例之1-同进程下直接继承Service

    Extending the Binder class If your service is used only by the local application and does not need t ...

  7. TestNG基本注解(二)

    1. Before类别和After类别注解 @BeforeSuite @AfterSuite @BeforeTest @AfterTest @BeforeClass @AfterClass @Befo ...

  8. poj2661Factstone Benchmark

    链接 利用log函数来求解 n!<=2^k k会达到400+W 暴力就不要想了,不过可以利用log函数来做 log2(n!) = log2(1)+log2(2)+..log2(n)<=k ...

  9. js 将XML字符串解析成XML文档 --- attribute construct error--- 空白字符与空格问题

    最近在做xml在线编辑器,遇到一个字符串解析成xml文档的问题,记录一下. 原始xml内容读取自xml文档 <label class="test" id="labe ...

  10. 高仿人人网客户端Android版项目源码

    高仿人人网客户端,有兴趣的盆友可以研究下,里面主要包含的一些UI设计与交互.(注:项目中有少许问题,apk能运行,希望开发者可以参考代码研究一下.) 源码下载:http://code.662p.com ...