• 避免在 PL/SQL 中使用嵌套游标查询


    考虑下面的 PL/SQL 代码,这段代码生成一个 XML 格式的矩阵样式的报表:
     declare
      l_count   
    integer;
      
    begin
      dbms_output.put_line(
    '<matrix>');
      
    -- generate matrix of parts by country
      for part in (select id,description from parts order by description) loop
      dbms_output.put_line(
    '<row>');
      dbms_output.put_line(
    '<cell>'||part.description||'</cell>');
      
    for country in (select code from countries order by name) loop
      
    select sum(cnt) into l_count from orders
      
    where part_id = part.id and cc = country.code;
      dbms_output.put_line(
    '<cell>'||nvl(l_count,0)||'</cell>');
      
    end loop;
      dbms_output.put_line(
    '</row>');
      
    end loop;
      dbms_output.put_line(
    '</matrix>');
      
    end;

      

    如果在这个例子中 parts 和 countries 有很多行数据,那么性能就会趋于下降。这是因为,在 PL/SQL 中,每次遇到一个游标 FOR 循环,在重新查询并获得数据时,都会有一个切换到 SQL 的上下文切换。

      以一些服务器端内存为代价,提高这种构造的速度是有可能做到的——如果动态构建 PL/SQL 数据表和矩阵单元格条目就可以提高速度。例如:
    declare
      type part_tbl_type 
    is table of parts%rowtype index by binary_integer;
      part_tbl   part_tbl_type;
      
    --
      type country_tbl_type is table of countries%rowtype index by binary_integer;
      country_tbl   country_tbl_type;
      
    --
      type cell_rec is record
      (
      part_id     orders.part_id
    %type,
      cc        orders.cc
    %type,
      cnt        orders.cnt
    %type
      );
      type cell_tbl_type 
    is table of cell_rec index by binary_integer;
      cell_tbl cell_tbl_type;
      
    --
      i pls_integer;
      
    begin
      
    -- build rows
      for row in (select * from parts order by description) loop
      part_tbl(part_tbl.
    count+1) := row;
      
    end loop;
      
    -- build columns
      for col in (select * from countries order by name) loop
      country_tbl(country_tbl.
    count+1) := col;
      
    end loop;
      
    -- build cells
      for cell in (select part_id,cc,sum(cnt) from orders group by part_id,cc) loop
      cell_tbl(cell_tbl.
    count+1) := cell;
      
    end loop;
      dbms_output.put_line(
    '<matrix>');
      
    -- generate matrix of parts by country
      i := cell_tbl.first;
      
    for row in part_tbl.first .. part_tbl.last loop
      dbms_output.put_line(
    '<row>');
      dbms_output.put_line(
    '<cell>'||part_tbl(row).description||'</cell>');
      
    for col in country_tbl.first .. country_tbl.last loop
      
    if cell_tbl(i).part_id = part_tbl(row).id
      
    and cell_tbl(i).cc = country_tbl(col).code
      
    then
      dbms_output.put_line(
    '<cell>'||cell_tbl(i).cnt||'</cell>');
      i :
    = i + 1;
      
    else
      dbms_output.put_line(
    '<cell>0</cell>');
      
    end if;
      
    end loop;
      dbms_output.put_line(
    '</row>');
      
    end loop;
      dbms_output.put_line(
    '</matrix>');
      
    end;


             

                成长

           /      |     \

        学习   总结   分享

    QQ交流群:122230156

  • 相关阅读:
    Oracle执行查询报错ORA-01034: ORACLE not available
    shell echo单行和多行文字定向写入到文件中
    oracle查看EM管理器状态显示Environment variable ORACLE_UNQNAME not defined. Please set ORACLE_UNQNAME to database unique name.
    正确使用 Element $confirm 方法,联调删除接口
    Express 统一配置响应头 header 方法
    jquery.i18n.properties.js 实现多语言
    navicat导出数据结构到word
    RabbitMQ用户角色及权限控制
    chrome谷歌浏览器自动填充用户名密码错位
    JVM性能调优参数整理
  • 原文地址:https://www.cnblogs.com/benio/p/1913695.html
Copyright © 2020-2023  润新知