使用USE_CONCAT提示[转帖]_MySQL, Oracle及数据库讨论区_Weblogic技术|Tuxedo技术|中间件技术|Oracle论坛|JAVA论坛|Linux/Unix技术|hadoop论坛_联动北方技术论坛  
网站首页 | 关于我们 | 服务中心 | 经验交流 | 公司荣誉 | 成功案例 | 合作伙伴 | 联系我们 |
联动北方-国内领先的云技术服务提供商
»  游客             当前位置:  论坛首页 »  自由讨论区 »  MySQL, Oracle及数据库讨论区 »
总帖数
1
每页帖数
101/1页1
返回列表
0
发起投票  发起投票 发新帖子
查看: 3709 | 回复: 0   主题: 使用USE_CONCAT提示[转帖]        下一篇 
wayne
注册用户
等级:中校
经验:1690
发帖:221
精华:0
注册:2011-7-21
状态:离线
发送短消息息给wayne 加好友    发送短消息息给wayne 发消息
发表于: IP:您无权察看 2011-7-24 19:02:46 | [全部帖] [楼主帖] 楼主

USE_CONCAT提示强迫优化器扩展查询中的每一个OR谓词为独立的查询块.
最后合并所有查询块的结果,返回结果集给用户。

当使用多个in-lists查询时,Oracle可能选择把单个查询扩展为多个查询块。

使用USE_CONCAT提示示例:

1.使用scott用户及标准表进行测试

$ sqlplus scott/tiger

SQL*Plus: Release 9.2.0.4.0 - Production on Wed Nov 17 15:17:51 2004

Copyright (c) 1982, 2002, Oracle Corporation. All rights reserved.

Connected to:
Oracle9i Enterprise Edition Release 9.2.0.4.0 - 64bit Production
With the Partitioning, OLAP and Oracle Data Mining options
JServer Release 9.2.0.4.0 - Production

SQL> set autotrace on
SQL> select * from emp where empno in (7788,7900);

EMPNO ENAME JOB MGR HIREDATE SAL COMM DEPTNO
---------- ---------- --------- ---------- --------- ---------- ---------- ----------
7788 SCOTT ANALYST 7566 19-APR-87 3000 20
7900 JAMES CLERK 7698 03-DEC-81 950 30

Execution Plan
----------------------------------------------------------
0 SELECT STATEMENT Optimizer=CHOOSE (Cost=2 Card=2 Bytes=74)
1 0 TABLE ACCESS (FULL) OF 'EMP' (Cost=2 Card=2 Bytes=74)

--注意,此处Oracle选择了全表扫描,因为成本较低。

Statistics
----------------------------------------------------------
0 recursive calls
0 db block gets
4 consistent gets
0 physical reads
0 redo size
1032 bytes sent via SQL*Net to client
655 bytes received via SQL*Net from client
2 SQL*Net roundtrips to/from client
0 sorts (memory)
0 sorts (disk)
2 rows processed

2.添加提示

SQL> select /*+ use_concat */ * from emp where empno in (7788,7900);

EMPNO ENAME JOB MGR HIREDATE SAL COMM DEPTNO
---------- ---------- --------- ---------- --------- ---------- ---------- ----------
7900 JAMES CLERK 7698 03-DEC-81 950 30
7788 SCOTT ANALYST 7566 19-APR-87 3000 20

Execution Plan
----------------------------------------------------------
0 SELECT STATEMENT Optimizer=CHOOSE (Cost=4 Card=2 Bytes=74)
1 0 CONCATENATION
2 1 TABLE ACCESS (BY INDEX ROWID) OF 'EMP' (Cost=2 Card=1 Bytes=37)
3 2 INDEX (UNIQUE SCAN) OF 'PK_EMP' (UNIQUE) (Cost=1 Card=14)
4 1 TABLE ACCESS (BY INDEX ROWID) OF 'EMP' (Cost=2 Card=1 Bytes=37)
5 4 INDEX (UNIQUE SCAN) OF 'PK_EMP' (UNIQUE) (Cost=1 Card=14)

--使用use_concat提示以后,Oracle将in-lists条件展开为两个查询块,分别使用索引,最后CONCATENATION得到最后输出。
--注意,这里强制使用索引导致成本上升为4。

Statistics
----------------------------------------------------------
0 recursive calls
0 db block gets
4 consistent gets
0 physical reads
0 redo size
1032 bytes sent via SQL*Net to client
655 bytes received via SQL*Net from client
2 SQL*Net roundtrips to/from client
0 sorts (memory)
0 sorts (disk)
2 rows processed

SQL>

3.Oracle对于执行计划的改写

对于inlist查询,Oracle通常会进行改写,将形如

select ..... from ....... where ....in (..........)


的sql语句,改写为union all的形式来执行,这个改写通常是潜在的。

然而这一改写可能存在问题,如果inlist中的值比较多的话,CBO花在分析执行路径上的时间和成本都会相当大,此时我们通常需要阻止Oracle的这一展开操作.
我们可以通过NO_EXPAND提示来阻止Oracle进行这样的改写。

那么实际上,在这里,USE_CONCAT和NO_EXPAND成了互为"反函数"。在使用了NO_EXPAND提示后,从Oracle8之后,Oracle会使用"inlist iterator"
方式来执行SQL,这样可以用到index。




赞(0)    操作        顶端 
总帖数
1
每页帖数
101/1页1
返回列表
发新帖子
请输入验证码: 点击刷新验证码
您需要登录后才可以回帖 登录 | 注册
技术讨论