读书笔记 《精通Oracle PL/SQL》

来源:互联网 发布:大数据在医疗中的应用 编辑:程序博客网 时间:2024/04/29 11:08

Mastering Oracle PL/SQL Practical Solutions

五位作者:

Connnor MaDonald

Chaim Katz

Christopher Beck

Joel R. Kallman

David C. Knox

 

安装

 

一、高效能的PL/SQL

这里故意避开术语“性能”,因为效能不只是性能。

 

效能是指什么呢?假定你的代码符合功能需求,则一般认为,PL/SQL代码满足以下3个条件才能称为高效能(efficient)。

a. 性能:运行的时间在可接受的范围内。

b. 影响:它不会破坏(或者以某种不可接受的方式削弱)你系统中任何其他组件。

c. 可论证性:你可以证明(或表明),在所有合理的条件下,代码都不会破坏上述性能和影响条件。简言之,即使你想破坏性能也做不到。

 

要使程序达到高效能,运行消耗的时间必须在要求的时间内,且不会对系统的其他部分或资源造成严重破坏。我们还要能够论效能在大多数或所有可预期的条件下可以保持。

我们在此只围绕达到高效能集中讨论3点大的准则。

a. 当解析所要执行的SQL代码时,尽量减少数据库的工作量。从某种意义上说,这不是特指PL/SQL,因为不只是PL/SQL运行SQL。它适用于执行SQL的任何语言(Java, Visual Baisc等)。然而,它可能是导致Oracle应用程序不可扩展的首要原因,因此需要加以讨论。

b. 理解PL/SQL的特征和函数,并掌握如何正确地开发以避免无谓地重复。

c. 绝不使用PL/SQ来做SQL的工作。

    文中列举了三个例子。

    1. 显示日历 

    2. 计算中间数 用解析函数percentile_count。

    3. 条件表的查找  用标量子查询功能。

  

二、全部打包

平时也用包,也就是将PL/SQL各部分组合在一起而已。

包的基本好处:

1. 包的重载

2. 包中的公有变量和私有变量

3. 初始化

4. 信息隐藏

 

信息隐藏的好处、永久的公有和私有变量以及初始化块,都不可避免地与包被分割成公有和私有部分联系在一起。该分割还提供了包对于独立过程的最有意义的优势:从所谓的“依赖危机(dependency crisis)”中独立出来。

 

 

三、令人困惑的游标

 

 游标是十分简单的事物。你让数据库准备些结果,之后你得到了这些结果,然后你告诉数据库这些结果用过了。

 

1. 显式游标与隐式游标

隐式游标的原理:

a. 打开游标

b. 读取行

c. 再次读取(以检查 TOO_MANY_ROWS)

d. 关闭游标

由于PL/SQL开发者能完全控制显式游标,可以编写代码只读取一次,因此显式游标更高效。这确实是一个令人信服的论据,在PL/SQL的早期版本中,这个论据曾经是有价值的。然而,早在Oracle版本7.1时,就引入了欲取的概念。简而言之,当调用欲取操作用于从游标中读取一行时,下一行也在同一调用中获得。集合现代磁盘驱动器和文件系统中的预先读取(read-ahead)类似。数据库预测,你通常都会获取下一行。

 

有个“存在性检查”的例子:检查是否有员工是上个月聘用的。

差劲的SQL:

select count(*) from emp whre hiredate > trunc(sysdate, 'MM');

 

较好:

select count(*) from dual where exists (select null from emp where hiredate > trunc(sysdate, 'MM'));

 

其实隐式游标并没有比显式游标更强大的神奇属性。然而,这里所证明的是,隐式游标在代码上更加简洁,且在大多数情况下,性能等同于等价的显式游标,甚至比它更强。

另外,隐式游标更容易理解,不容易想显式游标那样被误用。

 

2. 跨架构的游标管理

a. 游标变量

    游标变量提供了统一的方法,通过这种方法,查询结果集可以在应用程序组件之间传递。

    游标变量在客户端定义,在服务器端打开,而从客户端读取数据。不需要在服务器端创建结果集并将整个结果集传递回客户端,客户端能直接访问该游标。无论何处,只要你拥有支持游标变量的客户端应用程序(Java, Proc*C等),客户端程序就可以访问游标结果集,而不需要获得整个结果集,然后保存,并传回。


 

四、高效数据处理

 

在PL/SQL中使用集合

使用集合最主要的动机是,它鼓励开发者多以集的方式思考,而不是行。虽然没有功能上的原因阻止开发者使用每次处理结果集中的一行,但从高效和性能的角度来讲,这样并不好。

 

批收集: bulk collect

批量绑定: forall。它被设计用来减少另一种开销---PL/SQL内的上下文切换(context switching)。在PL/SQL程序启动后,数据库中的PL/SQL运行时引擎开始执行代码。当在代码中遇到SQL语句后,PL/SQL运行时引擎将SQL传递给SQL执行引擎并等待响应。所以,一般的PL/SQL块在运行时会在两个引擎之间前后跳来跳去。

    批量绑定的错误处理: save exceptions。现在当异常发生时,可以访问我们需要的信息来查明是什么导致了错误,更重要的是,可以取得集合中的行索引。sql%bulk_exceptions

    用limit子句。

 

 

五、PL/SQL优化技巧

1. 尽量减少解析和内存消耗

定义者权限(默认)

调用者权限

使用管道函数

 

2. 数据类型:提示和技巧

关联数组,集合,使用基于记录的DML时的问题

 

3.  调用PL/SQL

 

六、触发器

 七、DBA包

 八、安全包

 九、Web包

 十、PL/SQL调试

自定义的DEBUG工具

开发DEBUG包背后的主要思想是让开发者更好地控制谁产生消息以及产生什么消息,而不需修改应用程序。

基于该主要思想,DEBUG工具的核心需求如下:

  1. 对任何PL/SQL都可用

  2. 生成的调试数量无限制

  3. 生成的每行调试信息的长度无限制

  4. 实时反馈

  5. 自知(如,它知道自己是在哪个包中的哪一行)

  6. 能轻易地开启和关闭

  7. 能选择对包/过程/函数进行调试

  8. 根据运行的用户,而采取不同的行为

  9. 将消息保存于文件中

  10. 容易使用

 

既然代码可以从http://www.apress.com免费下载,我就顺便把它贴出来吧。

三个文件: debug_ddl.sql,debug.pkh,debug.sql。

 

============ debug_ddl.sql

drop table debugTab
/

create table debugTab(
  userid           varchar2(30),
  dir              varchar2(32),
  filename         varchar2(1024),
  modules          varchar2(4000),
  show_date        varchar2(3),
  date_format      varchar2(255),
  name_length      number,
  show_session_id  varchar2(3),
  --
  -- Constraints
  --
  constraint debugtab_pk
    primary key ( userid, filename ),
  constraint debugtab_show_date_ck
    check ( show_date in ( 'YES', 'NO' ) ),
  constraint debugtab_show_session_id_ck
    check ( show_session_id in ( 'YES', 'NO' ) )
)
/

create or replace
trigger biu_fer_debugtab
before insert or update on debugtab for each row
begin

  :new.modules := upper( :new.modules );
  :new.show_date := upper( :new.show_date );
  :new.show_session_id := upper( :new.show_session_id );
  :new.userid := upper( :new.userid );

  declare
    l_date varchar2(100);
  begin
    l_date := to_char( sysdate, :new.date_format );
  exception
    when others then
      raise_application_error(
        -20001,
        'Invalid date format "' ||
        :new.date_format || '"' );
  end;

  declare
    l_handle utl_file.file_type;
    --l_file varchar2(32767);
    --l_location varchar2(23767);
  begin
    --l_file := substr( :new.filename,
                      --instr( replace( :new.filename, '\', '/' ),
                                                     --'/', -1 )+1 );
    --l_location := substr( :new.filename,
                          --1,
                          --instr( replace( :new.filename, '\', '/' ),
                                                         --'/', -1 )-1 );
    l_handle := utl_file.fopen(
                  location => :new.dir,
                  filename => :new.filename,
                  open_mode => 'a',
                  max_linesize => 32767 );
    utl_file.fclose( l_handle );
  exception
    when others then
      raise_application_error(
        -20001,
        'Cannot open debug dir/file ' ||
        :new.dir || '/' ||
        :new.filename );
  end;

end;
/

 

============ debug.pkh

create or replace
package debug as

  --
  -- Type Definitions
  --

  type Argv is table of varchar2(4000);
  emptyDebugArgv Argv;

  --
  --  Initializes the debuging for specified p_modules and will dump the
  --  output to the p_dir directory on the server for the user p_user.
  --
  procedure init(
    p_modules     in varchar2 default 'ALL',
    p_dir         in varchar2 default 'TEMP',
    p_file        in varchar2 default user || '.dbg',
    p_user        in varchar2 default user,
    p_show_date   in varchar2 default 'YES',
    p_date_format in varchar2 default 'MMDDYYYY HH24MISS',
    p_name_len    in number default 30,
    p_show_sesid  in varchar2 default 'NO' );

  procedure status(
    p_user in varchar2 default user,
    p_dir  in varchar2 default null,
    p_file in varchar2 default null );

  procedure f(
    p_message in varchar2,
    p_arg1    in varchar2 default null,
    p_arg2    in varchar2 default null,
    p_arg3    in varchar2 default null,
    p_arg4    in varchar2 default null,
    p_arg5    in varchar2 default null,
    p_arg6    in varchar2 default null,
    p_arg7    in varchar2 default null,
    p_arg8    in varchar2 default null,
    p_arg9    in varchar2 default null,
    p_arg10   in varchar2 default null );

  procedure fa(
    p_message in varchar2,
    p_args    in Argv default emptyDebugArgv );

  procedure clear(
    p_user in varchar2 default user,
    p_dir  in varchar2 default null,
    p_file in varchar2 default null );

end debug;
/

show error

 

 

============ debug.sql

create or replace
package body debug as

  g_session_id varchar2(2000);

  procedure who_called_me(
    o_owner  in out varchar2,
    o_object    out varchar2,
    o_lineno    out number ) is
  --
    l_call_stack long default dbms_utility.format_call_stack;
    l_line varchar2(4000);
  begin

    /*
      ----- PL/SQL Call Stack -----
        object      line  object
        handle    number  name
      86c60290        17  package body OPS$CLBECK.DEBUG
      86c60290       212  package body OPS$CLBECK.DEBUG
      86c60290       251  package body OPS$CLBECK.DEBUG
      86aa28f0         1  procedure OPS$CLBECK.A
      86a9e940         1  anonymous block
    */

    -- skip three header lines and first levels in the stack
    for i in 1 .. 6 loop
      l_call_stack := substr( l_call_stack, instr( l_call_stack, chr(10) )+1 );
    end loop;

    -- set l_line to the current line
    l_line := substr( l_call_stack, 1, instr( l_call_stack, chr(10) ) - 1 );

    -- strip object handle
    l_line := ltrim( substr( l_line, instr( l_line, ' ' )));

    -- assign line number
    o_lineno := to_number(substr( l_line, 1, instr( l_line, ' ' )));
    l_line := ltrim( substr( l_line, instr( l_line, ' ' )));

    -- strip out object type
    l_line := ltrim( substr( l_line, instr( l_line, ' ' )));

    -- if 'package body' or 'anonymous block', strip out second piece
    if l_line like 'block%' or
       l_line like 'body%' then
      l_line := ltrim( substr( l_line, instr( l_line, ' ' )));
    end if;

    -- assign owner and object name
    o_owner := ltrim(rtrim(substr( l_line, 1, instr( l_line, '.' )-1 )));
    o_object  := ltrim(rtrim(substr( l_line, instr( l_line, '.' )+1 )));

    if o_owner is null then
      o_owner := user;
      o_object := 'ANONYMOUS BLOCK';
    end if;

  end who_called_me;

  function parse_it(
    p_message       in varchar2,
    p_argv          in argv,
    p_header_length in number ) return varchar2 is
  --
    l_message long := null;
    l_str long := p_message;
    l_idx number := 1;
    l_ptr number := 1;
  begin

    if nvl( instr( p_message, '%' ), 0 ) = 0 and
       nvl( instr( p_message, '\' ), 0 ) = 0 then
      return p_message;
    end if;

    loop

      l_ptr := instr( l_str, '%' );
      exit when l_ptr = 0 or l_ptr is null;
      l_message := l_message || substr( l_str, 1, l_ptr-1 );
      l_str :=  substr( l_str, l_ptr+1 );

      if substr( l_str, 1, 1 ) = 's' then
        l_message := l_message || p_argv(l_idx);
        l_idx := l_idx + 1;
        l_str := substr( l_str, 2 );

      elsif substr( l_str,1,1 ) = '%' then
        l_message := l_message || '%';
        l_str := substr( l_str, 2 );

      else
        l_message := l_message || '%';
      end if;

    end loop;

    l_str := l_message || l_str;
    l_message := null;

    loop

      l_ptr := instr( l_str, '\' );
      exit when l_ptr = 0 or l_ptr is null;
      l_message := l_message || substr( l_str, 1, l_ptr-1 );
      l_str :=  substr( l_str, l_ptr+1 );

      if substr( l_str, 1, 1 ) = 'n' then
        l_message := l_message || chr(10) ||
                     rpad( ' ', p_header_length, ' ' );
        l_str := substr( l_str, 2 );

      elsif substr( l_str, 1, 1 ) = 't' then
        l_message := l_message || chr(9);
        l_str := substr( l_str, 2 );

      elsif substr( l_str, 1, 1 ) = '\' then
        l_message := l_message || '\';
        l_str := substr( l_str, 2 );

      else
        l_message := l_message || '\';
      end if;

    end loop;

    return l_message || l_str;

  end parse_it;


  function build_it(
    p_debug_row in debugTab%rowtype,
    p_owner     in varchar2,
    p_object    in varchar2,
    p_lineno number ) return varchar2 is
  --
    l_header long := null;
  begin

    if p_debug_row.show_session_id = 'YES' then
      l_header := g_session_id || ' - ';
    end if;

    if p_debug_row.show_date = 'YES' then
      l_header := l_header ||
                  to_char( sysdate,
                           nvl( p_debug_row.date_format,
                                'MMDDYYYY HH24MISS' ) );
    end if;

    l_header :=
      l_header ||
      '(' ||
      lpad( substr( p_owner || '.' || p_object,
                    greatest( 1, length( p_owner || '.' || p_object ) -
                      least( p_debug_row.name_length, 61 ) + 1 ) ),
                    least( p_debug_row.name_length, 61 ) ) ||
      lpad( p_lineno,5 ) ||
      ') ';

    return l_header;

  end build_it;


  function file_it(
    p_dir     in debugtab.dir%type,
    p_file    in debugtab.filename%type,
    p_message in varchar2 ) return boolean is
  --
    l_handle utl_file.file_type;
    --l_file long;
    --l_location long;
  begin

    --l_file := substr( p_file,
                      --instr( replace( p_file, '\', '/' ),
                             --'/', -1 )+1 );

    --l_location := substr( p_file,
                          --1,
                          --instr( replace( p_file, '\', '/' ),
                                 --'/', -1 )-1 );

    l_handle := utl_file.fopen(
                  location => p_dir,
                  filename => p_file,
                  open_mode => 'a',
                  max_linesize => 32767 );

    utl_file.put( l_handle, '' );
    utl_file.put_line( l_handle, p_message );
    utl_file.fclose( l_handle );

    return true;

  exception
    when others then
      if utl_file.is_open( l_handle ) then
        utl_file.fclose( l_handle );
      end if;

      return false;

  end file_it;


  procedure debug_it(
    p_message in varchar2,
    p_argv    in argv ) is
  --
    l_message long := null;
    l_header long := null;
    call_who_called_me boolean := true;
    l_owner varchar2(255);
    l_object varchar2(255);
    l_lineno number;
    l_dummy boolean;
  begin

    for c in ( select *
                 from debugtab
                where userid = user )
    loop

      if call_who_called_me then
        who_called_me( l_owner, l_object, l_lineno );
        call_who_called_me := false;
      end if;

      if instr( ',' || c.modules || ',',
                ',' || l_object || ',' ) <> 0 or
         c.modules = 'ALL'
      then

        l_header := build_it( c, l_owner, l_object, l_lineno );
        l_message := parse_it( p_message, p_argv, length(l_header) );
        l_dummy := file_it( c.dir, c.filename, l_header || l_message );

      end if;
    end loop;

  end debug_it;


  --
  --  Public Procedures/Functions
  --

  procedure fa(
    p_message in varchar2,
    p_args    in Argv default emptyDebugArgv ) is
  begin
    debug_it( p_message, p_args );
  end fa;

  procedure f(
    p_message in varchar2,
    p_arg1    in varchar2 default null,
    p_arg2    in varchar2 default null,
    p_arg3    in varchar2 default null,
    p_arg4    in varchar2 default null,
    p_arg5    in varchar2 default null,
    p_arg6    in varchar2 default null,
    p_arg7    in varchar2 default null,
    p_arg8    in varchar2 default null,
    p_arg9    in varchar2 default null,
    p_arg10   in varchar2 default null ) is
  begin
    debug_it( p_message,
              argv( substr( p_arg1, 1, 4000 ),
                    substr( p_arg2, 1, 4000 ),
                    substr( p_arg3, 1, 4000 ),
                    substr( p_arg4, 1, 4000 ),
                    substr( p_arg5, 1, 4000 ),
                    substr( p_arg6, 1, 4000 ),
                    substr( p_arg7, 1, 4000 ),
                    substr( p_arg8, 1, 4000 ),
                    substr( p_arg9, 1, 4000 ),
                    substr( p_arg10, 1, 4000 ) ) );
  end f;


  procedure status(
    p_user in varchar2 default user,
    p_dir  in varchar2 default null,
    p_file in varchar2 default null ) is
  --
    l_found boolean := false;
  begin

    dbms_output.put_line( chr(10) );
    dbms_output.put_line( 'Debug info for ' ||
                          p_user );
    for c in ( select *
                 from debugtab
                where userid = p_user
                  and nvl( p_file, filename ) = filename
                  and nvl( p_dir, dir ) = dir )
    loop
      dbms_output.put_line( '---------------' ||
                            rpad( '-', length( p_user ), '-' ) );
      l_found := true;
      dbms_output.put_line( 'USER:                 ' ||
                            c.userid );
      dbms_output.put_line( 'MODULES:              ' ||
                            c.modules );
      dbms_output.put_line( 'DIRECTORY:            ' ||
                            c.dir );
      dbms_output.put_line( 'FILENAME:             ' ||
                            c.filename );
      dbms_output.put_line( 'SHOW DATE:            ' ||
                            c.show_date );
      dbms_output.put_line( 'DATE FORMAT:          ' ||
                            c.date_format );
      dbms_output.put_line( 'NAME LENGTH:          ' ||
                            c.name_length );
      dbms_output.put_line( 'SHOW SESSION ID:      ' ||
                            c.show_session_id );
      dbms_output.put_line( ' ' );
    end loop;

    if not l_found then
      dbms_output.put_line( 'No debug setup.' );
    end if;

  end status;


  procedure clear( p_user in varchar2 default user,
                   p_dir  in varchar2 default null,
                   p_file in varchar2 default null ) is
    pragma autonomous_transaction;
  begin
    delete from debugTab
     where userid = p_user
       and filename = nvl( p_file, filename )
       and dir = nvl( p_dir, dir );
    commit;
  end clear;


  procedure init(
    p_modules     in varchar2 default 'ALL',
    p_dir         in varchar2 default 'TEMP',
    p_file        in varchar2 default user || '.dbg',
    p_user        in varchar2 default user,
    p_show_date   in varchar2 default 'YES',
    p_date_format in varchar2 default 'MMDDYYYY HH24MISS',
    p_name_len    in number   default 30,
    p_show_sesid  in varchar2 default 'NO' ) is
  --
    pragma autonomous_transaction;
    debugTab_rec debugTab%rowtype;
    l_message long;
  begin

    delete from debugTab
     where userid = p_user
       and filename = p_file;

    insert into debugTab(
      userid, modules, dir, filename, show_date,
      date_format, name_length, show_session_id )
    values (
      p_user, p_modules, p_dir, p_file, p_show_date,
      p_date_format, p_name_len, p_show_sesid )
    returning
      userid, modules, dir, filename, show_date,
      date_format, name_length, show_session_id
    into
      debugTab_rec.userid, debugTab_rec.modules, debugTab_rec.dir,
      debugTab_rec.filename, debugTab_rec.show_date,
      debugTab_rec.date_format, debugTab_rec.name_length,
      debugTab_rec.show_session_id;

    l_message := chr(10) || 'Debug parameters initialized on ' ||
                 to_char( sysdate, 'dd-MON-yyyy hh24:mi:ss' ) || chr(10);
    l_message := l_message || '           USER: ' ||
      debugTab_rec.userid || chr(10);
    l_message := l_message || '        MODULES: ' ||
      debugTab_rec.modules || chr(10);
    l_message := l_message || '      DIRECTORY: ' ||
      debugTab_rec.dir || chr(10);
    l_message := l_message || '       FILENAME: ' ||
      debugTab_rec.filename || chr(10);
    l_message := l_message || '      SHOW DATE: ' ||
      debugTab_rec.show_date || chr(10);
    l_message := l_message || '    DATE FORMAT: ' ||
      debugTab_rec.date_format || chr(10);
    l_message := l_message || '    NAME LENGTH: ' ||
      debugTab_rec.name_length || chr(10);
    l_message := l_message || 'SHOW SESSION ID: ' ||
      debugTab_rec.show_session_id || chr(10);

    if not file_it( debugTab_rec.dir, debugTab_rec.filename, l_message ) then
      rollback;
      raise_application_error(
        -20001,
        'Can not open directory - file: ' ||
        debugTab_rec.dir || ' - ' || debugTab_rec.filename );
    end if;

    commit;

  end init;

begin
  g_session_id := userenv('SESSIONID');
end debug;
/

show error


 

 

 

 

 

 

很多地方值得仔细阅读。留着以后慢慢边看边记录。

原创粉丝点击