Oracle--SQL程序优化案例一

下面是存储过程的一部分程序:

PROCEDURE SAP_MAN_ROUTING_SO (CITEM_ID    VARCHAR2,
                                 CSITE_ID    VARCHAR2,
                                 CTYPE       VARCHAR2)
   IS
      V_ROUTING   VARCHAR2 (40);
        BEGIN                                 
      IF (CITEM_ID NOT LIKE ‘%C%‘)
      THEN
         SELECT NVL (TRIM (RR.ROUTING_ID), ‘‘)
           INTO V_ROUTING
           FROM (SELECT MAPL.PLNNR || ‘_‘ || MAPL.WERKS ROUTING_ID
                   FROM SAP_MAPL_QMCUSTOMER MAPL
                  WHERE     MAPL.MATNR = CITEM_ID
                        AND MAPL.WERKS = CSITE_ID
                        AND EXISTS
                               (SELECT NULL
                                  FROM SAP_MAPL_QMCUSTOMER
                                 WHERE     SAP_MAPL_QMCUSTOMER.MATNR =
                                              CITEM_ID
                                       AND SAP_MAPL_QMCUSTOMER.WERKS =
                                              CSITE_ID)) RR;
      ELSIF (CITEM_ID LIKE ‘%C%‘)
      THEN
         SELECT NVL (TRIM (RR.ROUTING_ID), ‘‘)
           INTO V_ROUTING
           FROM (SELECT MAX (MAPL.PLNNR) ROUTING_ID
                   FROM SAP_MAPL_QMCUSTOMER MAPL
                  WHERE MAPL.MATNR = CITEM_ID AND MAPL.WERKS = CSITE_ID) RR;
      END IF;
 从代码上来看,改程序似乎没有什么缺陷,执行也相当顺畅,但有一天突然一直报

ORA-01403: 未找到任何数据ORA-06512: 在 "STG.SAP_SO_BOM_PROC", line 428
ORA-06512: 在 "STG.SAP_SO_BOM_PROC", line 1097

通过分析发现缺少对结果集行数做一个判断,如果返回的是0行数据,那就是没有值,没有值就会报错,

所以以下是对该程序进行判断:

PROCEDURE SAP_MAN_ROUTING_SO (CITEM_ID    VARCHAR2,
                                 CSITE_ID    VARCHAR2,
                                 CTYPE       VARCHAR2)
   IS
      V_ROUTING   VARCHAR2 (40);
      qty         number;
     BEGIN
       select count(*) into qty          
           FROM (SELECT MAPL.PLNNR || ‘_‘ || MAPL.WERKS ROUTING_ID
                   FROM SAP_MAPL_QMCUSTOMER MAPL
                  WHERE     MAPL.MATNR = CITEM_ID
                        AND MAPL.WERKS = CSITE_ID
                        AND EXISTS
                               (SELECT NULL
                                  FROM SAP_MAPL_QMCUSTOMER
                                 WHERE     SAP_MAPL_QMCUSTOMER.MATNR =
                                              CITEM_ID
                                       AND SAP_MAPL_QMCUSTOMER.WERKS =
                                              CSITE_ID)) RR;
      IF qty>0  then                                       
      IF (CITEM_ID NOT LIKE ‘%C%‘)
      THEN
         SELECT NVL (TRIM (RR.ROUTING_ID), ‘‘)
           INTO V_ROUTING
           FROM (SELECT MAPL.PLNNR || ‘_‘ || MAPL.WERKS ROUTING_ID
                   FROM SAP_MAPL_QMCUSTOMER MAPL
                  WHERE     MAPL.MATNR = CITEM_ID
                        AND MAPL.WERKS = CSITE_ID
                        AND EXISTS
                               (SELECT NULL
                                  FROM SAP_MAPL_QMCUSTOMER
                                 WHERE     SAP_MAPL_QMCUSTOMER.MATNR =
                                              CITEM_ID
                                       AND SAP_MAPL_QMCUSTOMER.WERKS =
                                              CSITE_ID)) RR;
      ELSIF (CITEM_ID LIKE ‘%C%‘)
      THEN
         SELECT NVL (TRIM (RR.ROUTING_ID), ‘‘)
           INTO V_ROUTING
           FROM (SELECT MAX (MAPL.PLNNR) ROUTING_ID
                   FROM SAP_MAPL_QMCUSTOMER MAPL
                  WHERE MAPL.MATNR = CITEM_ID AND MAPL.WERKS = CSITE_ID) RR;
      END IF;
      ELSE
        --程序漏洞,没有考虑返回值为0行的结果,现加判断参数qty作为返回行数的总数,大于1将执行原先的程序,否则直接赋空值 add by zhangguipeng20170216
        V_ROUTING:=‘‘;
        END IF;

时间: 2024-08-24 00:49:44

Oracle--SQL程序优化案例一的相关文章

Oracle SQL性能优化

转载自:http://www.cnblogs.com/rootq/archive/2008/11/17/1334727.html (1)      选择最有效率的表名顺序(只在基于规则的优化器中有效): ORACLE的解析器按照从右到左的顺序处理FROM子句中的表名,FROM子句中写在最后的表(基础表 driving table)将被最先处理,在FROM子句中包含多个表的情况下,你必须选择记录条数最少的表作为基础表.如果有3个以上的表连接查询, 那就需要选择交叉表(intersection ta

<转>Oracle SQL性能优化

原文链接:http://www.cnblogs.com/rootq/archive/2008/11/17/1334727.html (1)      选择最有效率的表名顺序(只在基于规则的优化器中有效): ORACLE的解析器按照从右到左的顺序处理FROM子句中的表名,FROM子句中写在最后的表(基础表 driving table)将被最先处理,在FROM子句中包含多个表的情况下,你必须选择记录条数最少的表作为基础表.如果有3个以上的表连接查询, 那就需要选择交叉表(intersection t

Oracle Sql语句优化

1.最高效的删除重复记录方法 (因为使用了ROWID)   例子:  DELETE FROM  EMP E  WHERE  E.ROWID > (SELECT MIN(X.ROWID) FROM  EMP X  WHERE  X.EMP_NO = E.EMP_NO); 2.在含有子查询的 SQL 语句中 , 要特别注意减少对表的查询   例子:  SELECT TAB_NAME FROM TABLES WHERE (TAB_NAME,DB_VER) = (SELECT TAB_NAME,DB_V

Oracle SQL性能优化系列

1. 选用适合的ORACLE优化器 ORACLE的优化器共有3种: a. RULE (基于规则) b. COST (基于成本) c. CHOOSE (选择性) 设置缺省的优化器,可以通过对init.ora文件中OPTIMIZER_MODE参数的各种声明,如RULE,COST,CHOOSE,ALL_ROWS,FIRST_ROWS . 你当然也在SQL句级或是会话(session)级对其进行覆盖. 为了使用基于成本的优化器(CBO, Cost-Based Optimizer) , 你必须经常运行an

SQL性能优化案例分析

这段时间做一个SQL性能优化的案例分析, 整理了一下过往的案例,发现一个比较有意思的,拿出来给大家分享. 这个项目是我在项目开展2期的时候才加入的, 之前一期是个金融内部信息门户, 里面有个功能是收集各个上市公司的财报, 然后做各种分析, 数据图表展示, 使用的人数并不多, 仅百人左右. 2期打算面向行外用户, 刚开始预计同时在线人数不超过50, 就以50访问用户/秒的性能测试, 结果在把1期的图表类数据展示响应基本在5分钟左右, 属于严重不可用, 说说我们的服务器配置, 有2台网站前端承载用户

Oracle SQL 性能优化技巧

Select语句完整的执行顺序: SQL Select语句完整的执行顺序: 1. from子句组装来自不同数据源的数据: 2.where子句基于指定的条件对记录行进行筛选: 3.group by子句将数据划分为多个分组: 4.使用聚集函数进行计算: 5.使用having子句筛选分组: 6.计算所有的表达式: 7. 使用order by对结果集进行排序 性能优化技巧 1.选用适合的ORACLE优化器      ORACLE的优化器共有3种 A.RULE (基于规则) b.COST (基于成本) c

Oracle SQL语句优化34条

非常好用的SQL语句优化34条 1)选择最有效率的表名顺序(只在基于规则的优化器中有效): ORACLE 的解析器按照从右到左的顺序处理FROM子句中的表名,FROM子句中写在最后的表(基础表 driving table)将被最先处理,在FROM子句中包含多个表的情况下,你必须选择记录条数最少的表作为基 础表.如果有3个以上的表连接查询, 那就需要选择交叉表(intersection table)作为基础表, 交叉表是指那个被其他表所引用的表. (2) WHERE子句中的连接顺序.: ORACL

Oracle SQL性能优化技巧大总结

(1) 选择最有效率的表名顺序(只在基于规则的优化器中有效):     Oracle的解析器按照从右到左的顺序处理FROM子句中的表名,FROM子句中写在最后的表(基础表 driving table)将被最先处理,在FROM子句中包含多个表的情况下,你必须选择记录条数最少的表作为基础表.假如有3个以上的表连接查询, 那就需要选择交叉表(intersection table)作为基础表, 交叉表是指那个被其他表所引用的表. (2) WHERE子句中的连接顺序.:     ORACLE采用自下而上的

Oracle SQL性能优化(转)

(1)      选择最有效率的表名顺序(只在基于规则的优化器中有效): ORACLE的解析器按照从右到左的顺序处理FROM子句中的表名,FROM子句中写在最后的表(基础表 driving table)将被最先处理,在FROM子句中包含多个表的情况下,你必须选择记录条数最少的表作为基础表.如果有3个以上的表连接查询, 那就需要选择交叉表(intersection table)作为基础表, 交叉表是指那个被其他表所引用的表. (2)      WHERE子句中的连接顺序.: ORACLE采用自下而

浅谈ORACLE SQL语句优化经验

(1) 选择最有效率的表名顺序(只在基于规则的seo/' target='_blank'>优化器中有效): ORACLE 的解析器按照从右到左的顺序处理FROM子句中的表名,FROM子句中写在最后的表(基础表 driving table)将被最先处理,在FROM子句中包含多个表的情况下,你必须选择记录条数最少的表作为基础表.如果有3个以上的表连接查询, 那就需要选择交叉表(intersection table)作为基础表, 交叉表是指那个被其他表所引用的表. (2) WHERE子句中的连接顺序.