开发者社区


C++/C | Java | .Net/Windows | Linux/Unix | 开源 | 其他
首页 - 编程
编程
全文搜索:   

[SQL] 避免在 PL/SQL 中使用嵌套游标查询
作者: Builder.com
2004-06-14 18:16:56



本文译自Builder.com,未经许可请勿转载

考虑下面的 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;

【下一页】
声明:
Builder.com.cn(原ZDNet China应用开发频道)原创文章版权所有,未经许可严禁转载,且不构成投资建议。
近期相关报道:
实用技术文档
J2me XML
C/C++ C#
Java Oracle
Mysql .Net
VB.NET CSS
SQL Server 数据库
SQL UNIX
Linux Jsp
PHP Perl
Javascript IIS
XHTML ColdFusion
ASP/ASP.NET Apache
AJAX
订阅技术邮件
订阅"技术圈"杂志!请在下面选择您感兴趣的专题,填写e-mail地址,然后按订阅按钮:
应用开发管理
VS.NET 周刊
Database 周刊
WEB Service周刊
JAVA 周刊
IT 认证
Windows服务器周刊
互联网开发
当Windows Server 2008专家得5000元现金大奖
CNET NETWORKS 中国: 爱卡汽车网 | CNET科技资讯网 | e询网 | CWEEK | 蜂鸟网 | GameSpot China | 个人电脑 | PChome | SPN | 网友世界 | ZDNet China | 中关村在线
CNET NETWORKS 美国: BNET | CNET.com | CNET Download.com | CNET News.com | CNET Reviews | CNET Shopper.com | GameSpot | MP3.com | mySimon | Release 1.0 | Search.com | TechRepublic | TV.com | Webshots | ZDNet
Copyright (c) 2006 CNET Networks 版权所有。 ZDNet 是CNET Networks公司注册服务商标。
ZDNet 公司标识是 CNET Networks公司注册服务商标。
中华人民共和国电信与信息服务业务经营许可证编号:京ICP证010391号