FIRST_ROWS優化模式語言排序模糊匹配問題

發表于:2009-08-04來源:作者:點擊數: 標簽:FIRSTROWSFirst模式語言
FIRST_ROWS優化模式語言排序模糊匹配問題 軟件測試工程師 標題比較長,不過只有這樣才能把問題描述清楚。 問題詳細描述為,在FIRST_ROWS優化模式下,將會話排序和比較方式設置為語義模式,即忽略大小寫模式,對字段進行LIKE模糊查詢,可能導致錯誤的結果。

FIRST_ROWS優化模式語言排序模糊匹配問題  軟件測試工程師

標題比較長,不過只有這樣才能把問題描述清楚。

問題詳細描述為,在FIRST_ROWS優化模式下,將會話排序和比較方式設置為語義模式,即忽略大小寫模式,對字段進行LIKE模糊查詢,可能導致錯誤的結果。

下面直接看問題的現象:

SQL> CREATE TABLE T1 (ID NUMBER PRIMARY KEY, NAME VARCHAR2(30));

表已創建。

SQL> CREATE INDEX IND_T1_NAME ON T1(NAME);

索引已創建。

SQL> INSERT INTO T1 SELECT ROWNUM, CHR(64 + ROWNUM) 
2 FROM ALL_OBJECTS WHERE ROWNUM <= 26;

已創建26行。

SQL> COMMIT;

提交完成。

SQL> ALTER SESSION SET NLS_COMP = LINGUISTIC;

會話已更改。

SQL> ALTER SESSION SET NLS_SORT = BINARY_CI;

會話已更改。

SQL> SELECT * FROM T1 WHERE NAME LIKE ’a%’;

ID NAME
---------- ------------------------------
1 A

SQL> SELECT /*+ FIRST_ROWS */ * FROM T1 WHERE NAME LIKE ’a%’;

未選定行

只要修改上面提到的關鍵點中的任意一個,就不會產生這個錯誤的現象:

SQL> SELECT /*+ ALL_ROWS */ * FROM T1 WHERE NAME LIKE ’a%’;

ID NAME
---------- ------------------------------
1 A

SQL> SELECT /*+ FIRST_ROWS */ * FROM T1 WHERE NAME = ’a’;

ID NAME
---------- ------------------------------
1 A

SQL> ALTER SESSION SET NLS_SORT = BINARY;

會話已更改。

SQL> ALTER SESSION SET NLS_COMP = BINARY;

會話已更改。

SQL> SELECT /*+ FIRST_ROWS */ * FROM T1 WHERE NAME LIKE ’A%’;

ID NAME
---------- ------------------------------
1 A

SQL> ALTER SESSION SET NLS_COMP = LINGUISTIC;

會話已更改。

SQL> ALTER SESSION SET NLS_SORT = BINARY_CI;

會話已更改。

SQL> SELECT /*+ FIRST_ROWS */ * FROM T1 WHERE NAME LIKE ’A%’;

ID NAME
---------- ------------------------------
1 A

SQL> SELECT /*+ FIRST_ROWS */ * FROM T1 WHERE NAME LIKE ’a’;

未選定行

通過上面的幾個查詢可以看到,問題和FIRST_ROWS,LIKE操作以及基于語義的排序直接相關,下面看看Oracle在異常情況下采用了何種執行計劃:

SQL> SET AUTOT ON EXP
SQL> SELECT /*+ FIRST_ROWS */ * FROM T1 WHERE NAME LIKE ’a’;

未選定行

執行計劃
----------------------------------------------------------
Plan hash value: 3350237141

-------------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
-------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 30 | 35 (0)| 00:00:01 |
|* 1 | VIEW | index$_join$_001 | 1 | 30 | 35 (0)| 00:00:01 |
|* 2 | HASH JOIN | | | | | |
|* 3 | INDEX RANGE SCAN | IND_T1_NAME | 1 | 30 | 3 (34)| 00:00:01 | [Page]
| 4 | INDEX FAST FULL SCAN| SYS_C006622 | 1 | 30 | 33 (0)| 00:00:01 |
-------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

1 - filter(\"NAME\" LIKE ’a’)
2 - aclearcase/" target="_blank" >ccess(ROWID=ROWID)
3 - access(\"NAME\" LIKE ’a’)

Note
-----
- dynamic sampling used for this statement

由于索引中并不包含語義查詢的結果,因此Oracle這里必須訪問表才能得到最終的結果,因此這個執行計劃是錯誤的:

SQL> SELECT * FROM T1 WHERE NAME = ’a’;

ID NAME
---------- ------------------------------
1 A

執行計劃
----------------------------------------------------------
Plan hash value: 3617692013
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
--------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 30 | 3 (0)| 00:00:01 |
|* 1 | TABLE ACCESS FULL| T1 | 1 | 30 | 3 (0)| 00:00:01 |
--------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

1 - filter(NLSSORT(\"NAME\",’nls_sort=’’BINARY_CI’’’)=HEXTORAW(’6100’)
)

Note
-----
- dynamic sampling used for this statement

SQL> SELECT /*+ INDEX(T1) */ * FROM T1 WHERE NAME = ’a’;

ID NAME
---------- ------------------------------
1 A

執行計劃
----------------------------------------------------------
Plan hash value: 159298173

-------------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
-------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 30 | 827 (1)| 00:00:10 |
|* 1 | TABLE ACCESS BY INDEX ROWID| T1 | 1 | 30 | 827 (1)| 00:00:10 |
| 2 | INDEX FULL SCAN | SYS_C006622 | 26 | | 26 (0)| 00:00:01 |
-------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

1 - filter(NLSSORT(\"NAME\",’nls_sort=’’BINARY_CI’’’)=HEXTORAW(’6100’) )

Note
-----
- dynamic sampling used for this statement

上面的兩個執行計劃已經說明了問題的關鍵,Oracle對于語義的排序無法通過索引獲取,必須要訪問表或者相應的函數索引,詳細描述可以參考文章開頭部分給出的鏈接。

而采用了FIRST_ROWS優化模式后,當操作為LIKE時,Oracle優化器選擇了錯誤的執行計劃進行了優化,采用索引的范圍掃描代替了表,從而引發了錯誤: [Page]

SQL> SELECT /*+ INDEX_JOIN(T1 IND_T1_NAME SYS_C006622) */ *
2 FROM T1
3 WHERE NAME LIKE ’a’;

未選定行

執行計劃
----------------------------------------------------------
Plan hash value: 3350237141

-------------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
-------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 30 | 35 (0)| 00:00:01 |
|* 1 | VIEW | index$_join$_001 | 1 | 30 | 35 (0)| 00:00:01 |
|* 2 | HASH JOIN | | | | | |
|* 3 | INDEX RANGE SCAN | IND_T1_NAME | 1 | 30 | 3 (34)| 00:00:01 |
| 4 | INDEX FAST FULL SCAN| SYS_C006622 | 1 | 30 | 33 (0)| 00:00:01 |
-------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

1 - filter(\"NAME\" LIKE ’a’)
2 - access(ROWID=ROWID)
3 - access(\"NAME\" LIKE ’a’)

Note
-----
- dynamic sampling used for this statement

現在沒有使用FIRST_ROWS,而采用HINT也達到了相同的效果。而解決這個問題的方法就是通過HINT來避免索引范圍掃描的發生。

SQL> ALTER SESSION SET OPTIMIZER_MODE = FIRST_ROWS;

會話已更改。

SQL> SELECT * FROM T1 WHERE NAME LIKE ’a’;

未選定行

執行計劃
----------------------------------------------------------
Plan hash value: 3350237141

-------------------------------------------------------------------------------------------

| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
-------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 30 | 35 (0)| 00:00:01 |
|* 1 | VIEW | index$_join$_001 | 1 | 30 | 35 (0)| 00:00:01 |
|* 2 | HASH JOIN | | | | | |
|* 3 | INDEX RANGE SCAN | IND_T1_NAME | 1 | 30 | 3 (34)| 00:00:01 |
| 4 | INDEX FAST FULL SCAN| SYS_C006622 | 1 | 30 | 33 (0)| 00:00:01 |
-------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

1 - filter(\"NAME\" LIKE ’a’)
2 - access(ROWID=ROWID)
3 - access(\"NAME\" LIKE ’a’)

Note
-----
- dynamic sampling used for this statement

SQL> SELECT /*+ FULL(T1) */ * FROM T1 WHERE NAME LIKE ’a’; [Page]

ID NAME
---------- ------------------------------
1 A

執行計劃
----------------------------------------------------------
Plan hash value: 3617692013

--------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
--------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 30 | 3 (0)| 00:00:01 |
|* 1 | TABLE ACCESS FULL| T1 | 1 | 30 | 3 (0)| 00:00:01 |
--------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

1 - filter(\"NAME\" LIKE ’a’)

Note
-----
- dynamic sampling used for this statement

SQL> SELECT /*+ NO_INDEX(T1) */ * FROM T1 WHERE NAME LIKE ’a’;

ID NAME
---------- ------------------------------
1 A

執行計劃
----------------------------------------------------------
Plan hash value: 3617692013

--------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
--------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 30 | 3 (0)| 00:00:01 |
|* 1 | TABLE ACCESS FULL| T1 | 1 | 30 | 3 (0)| 00:00:01 |
--------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

1 - filter(\"NAME\" LIKE ’a’)

Note
-----
- dynamic sampling used for this statement

查詢metalink,Oracle在Doc ID: Note:5252496.8明確說明了這個bug,這個bug會在Oracle10.2.0.4和11.1.0.6中被Fixed。

 

原文轉自:http://www.anti-gravitydesign.com

国产97人人超碰caoprom_尤物国产在线一区手机播放_精品国产一区二区三_色天使久久综合给合久久97