Oracle数据库Where条件执行顺序及Where子句的条件顺序对性能的影响

1、Oracle数据库Where条件执行顺序:

由于SQL优化起来比较复杂,并且还会受环境限制,在开发过程中,写SQL必须必须要遵循以下几点的原则:

1.ORACLE采用自下而上的顺序解析WHERE子句,根据这个原理,表之间的连接必须写在其他WHERE条件之前, 那些可以过滤掉最大数量记录的条件必须写在WHERE子句的末尾.

例如:

(低效)

SELECTFROM EMP E WHERE SAL > 50000 AND JOB = ‘MANAGER’ AND 25 < (SELECT COUNT(*) FROM EMP WHERE MGR=E.EMPNO);

(高效)

SELECTFROM EMP E WHERE 25 < (SELECT COUNT(*) FROM EMP WHERE MGR=E.EMPNO) AND SAL > 50000 AND JOB = ‘MANAGER’;

2.SELECT子句中避免使用’*’

当在SELECT子句中列出所有的COLUMN时,使用动态SQL列引用 ‘’ 是一个方便的方法.可是,这是一个非常低效的方法. 实际上,ORACLE在解析的过程中, 会将’’ 依次转换成所有的列名, 这个工作是通过查询数据字典完成的, 这意味着将耗费更多的时间.

3.使用表的别名(Alias)

当在SQL语句中连接多个表时, 请使用表的别名并把别名前缀于每个Column上.这样一来,就可以减少解析的时间并减少那些由Column歧义引起的语法错误.

注:Column歧义指的是由于SQL中不同的表具有相同的Column名,当SQL语句中出现这个Column时,SQL解析器无法判断这个Column的归属。

2、oracle中Where子句的条件顺序对性能的影响:

经常有人问到oracle中的Where子句的条件书写顺序是否对SQL性能有影响,我的直觉是没有影响,因为如果这个顺序有影响,Oracle应 该早就能够做到自动优化,但一直没有关于这方面的确凿证据。在网上查到的文章,一般认为在RBO优化器模式下无影响(10G开始,缺省为RBO优化器模 式),而在CBO优化器模式下有影响,主要有两种观点:

a.能使结果最少的条件放在最右边,SQL执行是按从右到左进行结果集的筛选的;

b.有人试验表明,能使结果最少的条件放在最左边,SQL性能更高。

查过oracle8到11G的在线文档,关于SQL优化相关章节,没有任何文档说过where子句中的条件对SQL性能有影响,到底哪种观点是 对的,没有一种确切的结论,只好自己来做实验证明。结果表明,SQL条件的执行是从右到左的,但条件的顺序对SQL性能没有影响。

实验一:证明了SQL的语法分析是从右到左的

下面的试验在9i和10G都可以得到相同的结果: 第1条语句执行不会出错,第2条语句会提示除数不能为零。

  1.Select 'ok' From Dual Where 1 / 0 = 1 And 1 = 2;

  2.Select 'ok' From Dual Where 1 = 2 And 1 / 0 = 1;

证明了SQL的语法分析是从右到左的。

实验二:证明了SQL条件的执行是从右到左的

  drop table temp; 
  create table temp( t1 varchar2(10),t2 varchar2(10)); 
  insert into temp values('zm','abcde'); 
  insert into temp values('sz','1'); 
  insert into temp values('sz','2'); 
  commit;
  1. select * from temp where to_number(t2)>1 and t1='sz';

  2. select * from temp where t1='sz' and to_number(t2)>1;

在9i上执行, 第1条语句执行不会出错,第2条语句会提示“无效的数字”

在10G上执行,两条语句都不会出错。

说明:9i上,SQL条件的执行确实是从右到左的,但是10G做了什么调整呢?

实验三:证明了在10g上SQL条件的执行是从右到左的

Create Or Replace Function F1(v_In Varchar2) Return Varchar2 Is 
  Begin 
  Dbms_Output.Put_Line('exec F1'); 
  Return v_In; 
  End F1; 
  / 
  Create Or Replace Function F2(v_In Varchar2) Return Varchar2 Is 
  Begin 
  Dbms_Output.Put_Line('exec F2'); 
  Return v_In; 
  End F2; 
  / 
  SQL> set serverout on; 
  SQL> select 1 from dual where f1('1')='1' and f2('1')='1'; 
  1 
  ---------- 
  1 
  exec F2 
  exec F1 
  SQL> select 1 from dual where f2('1')='1' and f1('1')='1'; 
  1 
  ---------- 
  1 
  exec F1 
  exec F2

结果表明,SQL条件的执行顺序是从右到左的。

那么,根据这个结果来分析,把能使结果最少的条件放在最右边,是否会减少其它条件执行时所用的记录数量,从而提高性能呢?

例如:下面的SQL条件,是否应该调整SQL条件的顺序呢?

  Where A.结帐id Is Not Null

  And A.记录状态<>0

  And A.记帐费用=1

  And (Nvl(A.实收金额, 0)<>Nvl(A.结帐金额, 0) Or Nvl(A.结帐金额, 0)=0)

  And A.病人ID=[1] And Instr([2],','||Nvl(A.主页ID,0)||',')>0

  And A.登记时间Between [3] And [4]

  And A.门诊标志<>1

实际上,从这条SQL语句的执行计划来分析,Oracle首先会找出条件中使用索引或表间连接的条件,以此来过滤数据集,然后对这些结果数据块所涉及的记录逐一检查是否符合所有条件,所以条件顺序对性能几乎没有影响。

Create Or Replace Function F1(v_In Varchar2) Return Varchar2 Is 
  Begin 
  Dbms_Output.Put_Line('exec F1'); 
  Return v_In; 
  End F1; 
  / 
  Create Or Replace Function F2(v_In Varchar2) Return Varchar2 Is 
  Begin 
  Dbms_Output.Put_Line('exec F2'); 
  Return v_In; 
  End F2; 
  / 
  SQL> set serverout on; 
  SQL> select 1 from dual where f1('1')='1' and f2('1')='1'; 
  1 
  ---------- 
  1 
  exec F2 
  exec F1 
  SQL> select 1 from dual where f2('1')='1' and f1('1')='1'; 
  1 
  ---------- 
  1 
  exec F1 
  exec F2

结果表明,SQL条件的执行顺序是从右到左的。

那么,根据这个结果来分析,把能使结果最少的条件放在最右边,是否会减少其它条件执行时所用的记录数量,从而提高性能呢?

例如:下面的SQL条件,是否应该调整SQL条件的顺序呢?

  Where A.结帐id Is Not Null

  And A.记录状态<>0

  And A.记帐费用=1

  And (Nvl(A.实收金额, 0)<>Nvl(A.结帐金额, 0) Or Nvl(A.结帐金额, 0)=0)

  And A.病人ID=[1] And Instr([2],','||Nvl(A.主页ID,0)||',')>0

  And A.登记时间Between [3] And [4]

  And A.门诊标志<>1

实际上,从这条SQL语句的执行计划来分析,Oracle首先会找出条件中使用索引或表间连接的条件,以此来过滤数据集,然后对这些结果数据块所涉及的记录逐一检查是否符合所有条件,所以条件顺序对性能几乎没有影响。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值