Archive

Archive for the ‘PL/SQL’ Category

PRAGMA EXCEPTION_INIT的用法

September 1st, 2011 No comments
 如果要处理未命名的内部异常,必须使用OTHERS异常处理器或PRAGMA EXCEPTION_INIT 。PRAGMA由编译器控制,或者是对于编译器的注释。PRAGMA在编译时处理,而不是在运行时处理。EXCEPTION_INIT告诉编译器将异常名与ORACLE错误码结合起来,这样可以通过名字引用任意的内部异常,并且可以通过名字为异常编写一适当的异常处理器。
  
  在子程序中使用EXCEPTION_INIT的语法如下:
  PRAGMA EXCEPTION_INIT(exception_name, -Oracle_error_number);
Read more…
Categories: PL/SQL Tags:

ORACLE PL/SQL异常处理(Exception)学习笔记

September 1st, 2011 No comments
1、PL/SQL错误类型
错误类型
报告者
处理方法
编译时错误
PL/SQL编译器
交互式地处理:编译器报告错误,你必须更正这些错误
运行时错误
PL/SQL运行时引擎
程序化地处理:异常由异常处理子程序引发并进行捕获
Read more…
Categories: PL/SQL Tags: , ,

Oracle数据库中null的使用详解

March 7th, 2009 No comments

  问:什么是NULL?

  答:在我们不知道具体有什么数据的时候,也即未知,可以用NULL,

  我们称它为空,ORACLE中,含有空值的表列长度为零。

  ORACLE允许任何一种数据类型的字段为空,除了以下两种情况:

  1、主键字段(primary key),

  2、定义时已经加了NOT NULL限制条件的字段

  说明:

  1、等价于没有任何值、是未知数。

  2、NULL与0、空字符串、空格都不同。

  3、对空值做加、减、乘、除等运算操作,结果仍为空。

  4、NULL的处理使用NVL函数。

  5、比较时使用关键字用“is null”和“is not null”。

  6、空值不能被索引,所以查询时有些符合条件的数据可能查不出来,

  count(*)中,用nvl(列名,0)处理后再查。

  7、排序时比其他数据都大(索引默认是降序排列,小→大),

  所以NULL值总是排在最后。

  使用方法:

SQL> select 1 from dual where null=null;
没有查到记录
SQL> select 1 from dual where null='';
没有查到记录
SQL> select 1 from dual where ''='';
没有查到记录
SQL> select 1 from dual where null is null;
        1
---------
        1
SQL> select 1 from dual where nvl(null,0)=nvl(null,0);
        1
---------
        1

  对空值做加、减、乘、除等运算操作,结果仍为空。

SQL> select 1+null from dual;
SQL> select 1-null from dual;
SQL> select 1*null from dual;
SQL> select 1/null from dual;

  查询到一个记录。

  注:这个记录就是SQL语句中的那个null

  设置某些列为空值

  update table1 set 列1=NULL where 列1 is not null;

  现有一个商品销售表sale,表结构为:

  month  char(6)  ——月份

  sellnumber(10,2) ——月销售金额

create table sale (month char(6),sell number);
insert into sale values('200001',1000);
insert into sale values('200002',1100);
insert into sale values('200003',1200);
insert into sale values('200004',1300);
insert into sale values('200005',1400);
insert into sale values('200006',1500);
insert into sale values('200007',1600);
insert into sale values('200101',1100);
insert into sale values('200202',1200);
insert into sale values('200301',1300);
insert into sale values('200008',1000);
insert into sale(month) values('200009');
         (注意:这条记录的sell值为空)
commit;

  共输入12条记录

SQL> select * from sale where sell like '%';
MONTH       SELL
------ ---------
200001      1000
200002      1100
200003      1200
200004      1300
200005      1400
200006      1500
200007      1600
200101      1100
200202      1200
200301      1300
200008      1000

  查询到11记录。

  结果说明:

  查询结果说明此SQL语句查询不出列值为NULL的字段

  此时需对字段为NULL的情况另外处理。

SQL> select * from sale where sell like '%' or sell is null;
SQL> select * from sale where nvl(sell,0) like '%';

MONTH       SELL
------ ---------
200001      1000
200002      1100
200003      1200
200004      1300
200005      1400
200006      1500
200007      1600
200101      1100
200202      1200
200301      1300
200008      1000
200009

  查询到12记录。

  Oracle的空值就是这么的用法,我们最好熟悉它的约定,以防查出的结果不正确。

  总结:null就是null,和别的类型没有可比性

Categories: oracle, PL/SQL Tags:

oracle 触发器全扫描

March 6th, 2009 No comments

 触发器
2.1 触发器的创建
CREATE TRIGGER [schema.]trigger_name
{BEFORE|AFTER} {UPDATE|INSERT|DELETE} ON [schema.]table_name
[ [REFERENCING correlation_names] FOR EACH ROW [WHEN (condition)] ]
DECLARE
declaration
BEGIN
pl/sql code
END;
/
2.2 触发器的管理
1) 查看触发器:
SQL> desc ALL_TRIGGERS;
名称 空? 类型
——————– ——– ———-
owner VARCHAR2(30)
trigger_name VARCHAR2(30)
trigger_type VARCHAR2(16)
triggering_event VARCHAR2(216)
table_owner VARCHAR2(30)
base_object_type VARCHAR2(16)
table_name VARCHAR2(30)
column_name VARCHAR2(4000)
referencing_names VARCHAR2(128)
when_clause VARCHAR2(4000)
status VARCHAR2(8)
description VARCHAR2(4000)
action_type VARCHAR2(11)
trigger_body LONG
2) 查看触发器的代码:
触发器的源代码被存贮在 ALL_TRIGGERS 的 trigger_body 字段中。抽取触发器定义的命令:
SET ECHO OFF
SET MAXDATA 20000
SET LONG 20000
SET LONGCHUNKSIZE 1000
SET PAGESIZE 0
SET HEADING OFF
SET TRIMSPOOL ON
SET TRIMOUT ON
SET RECSEP OFF
ACCEPT trigger_name CHAR PROMPT ‘please input the trigger to lookup:’
ACCEPT trigger_owner CHAR PROMPT ‘please input the trigger owner:’
ACCEPT file_name CHAR PROMPT ‘please enter the output file path:’
SET TERMOUT OFF
SET FEEDBACK OFF
– 设置when_clause字段格式并使之折行
COLUMN when_clause FORMAT A60 WORD_WRAPPED
SPOOL &file_name
SELECT ‘CREATE OR REPLACE TRIGGER ‘ || description FROM all_triggers
WHERE trigger_name = UPPER(‘&trigger_name’)
AND owner = UPPER(‘&trigger_owner’);
SELECT ‘WHEN (‘ || when_clause || ‘)’ when_caluse FROM all_triggers
WHERE trigger_name = UPPER(‘&trigger_name’)
AND owner = UPPER(‘&trigger_owner’)
AND when_clause IS NOT NULL;
SELECT trigger_body FROM all_triggers
WHERE trigger_name = UPPER(‘&trigger_name’)
AND owner = UPPER(‘&trigger_owner’);
SELECT ‘/’ FROM DUAL;
SPOOL OFF
SET TERMOUT ON
SET FEEDBACK ON
SET VERIFY ON
SET HEADING ON
SET PAGESIZE 24
3) 打开和关闭触发器:
– 打开触发器
ALTER TRIGGER trigger_name DISABLE;
– 关闭触发器
ALTER TRIGGER trigger_name ENABLE;
2.3 触发器的新功能
创建作用于特定数据库与数据定义事件的触发器。事件包括:
?? 关闭数据库(startup);
?? 启动数据库(shutdown);
?? 登陆数据库(logon);
?? 退出数据库(logoff);
?? 发生服务器错误(server error);
?? 创建对象(create命令);
?? 修改对象(alter命令);
?? 删除对象(delete命令);
1) 数据库事件触发器:
数据库事件触发器可以定义为数据库级和模式级。
要创建数据库级的触发器,必须有administrator database trigger 的系统权限,该权限提供
给DBA角色,通常只有数据库管理员有此权限。
事 件 触发器关键词 模 式 级 数据库级
启动数据库 STARTUP 否 是
关闭数据库 SHUTDOWN 否 是
服务器错误 SERVERERROR 否 是
登陆数据库 LOGON 是 是
退出数据库 LOGOFF 是 是
创建对象 CREATE 是 否
修改对象 ALTER 是 否
删除对象 DELETE 是 否
2) 事件属性:
编写触发器时,可以象变量一样使用这些属性。
事件属性 描 述 适应的触发器类型
sys.sysevent 返回一个20个字符的字符串,描述导致触发器
激发的事件 所有类型
sys.intance_num 返回Oracle 实例编号 所有类型
sys.database_name 返回数据库品牌,通常为”Oracle” 所有类型
sys.server_error
(stack_position)
从错误堆栈的指定位置返回错误号,使用
sys.server_error(1) 查找最近的错误 SERVERERROR
sys.login_user 返回导致触发器激发的用户名 所有类型
sys.dictionary_obj_
type
返回当DDL触发器激发时涉及到的对象类型 CREATE、ALTER、DROP
sys.dictionary_obj_
name
返回当DDL触发器激发时涉及到的对象名称 CREATE、ALTER、DROP
sys.des_encrypted_p
assword
创建或修改用户时,返回加密后的该用户的密
码 CREATE、ALTER、DROP
3) 创建数据库事件触发器实例:
CREATE TABLE uptime_log(
database_name VARCHAR2(30),
event_name VARCHAR2(30),
event_time DATE,
triggered_by_user VARCHAR2(30));
– 创建启动数据库时的事件触发器。
CREATE OR REPLACE TRIGGER log_startup
AFTER STARTUP ON DATABASE
BEGIN
INSERT INTO uptime_log VALUES(sys.database_name,
sys.sysevent, sysdate, sys.login_user);
END;
/
– 创建关闭数据库时的事件触发器。
CREATE OR REPLACE TRIGGER log_shutdown
AFTER SHUTDOWN ON DATABASE
BEGIN
INSERT INTO uptime_log VALUES(sys.database_name,
sys.sysevent, sysdate, sys.login_user);
END;
/
4) 创建DDL事件触发器实例:
CREATE TABLE alter_audit_trail(
object_owner VARCHAR2(30),
object_name VARCHAR2(30),
object_type VARCHAR2(30),
alter_by_user VARCHAR2(30),
sys_event VARCHAR2(30),
alteration_time DATE);
– 当SCOTT用户创建、修改或删除时的模式触发器。
CREATE OR REPLACE TRIGGER tri_alter_audit_trail
AFTER ALTER OR CREATE OR DROP ON scott.SCHEMA
BEGIN
IF sys.sysevent = ‘ALTER’ THEN
INSERT INTO alter_audit_trail VALUES
(sys.dictionary_obj_owner, sys.dictionary_obj_name,
sys.dictionary_obj_type, sys.login_user, ‘my_alter’, sysdate);
ELSIF sys.sysevent = ‘CREATE’ THEN
NULL;
ELSIF sys.sysevent = ‘DROP’ THEN
NULL;
END IF;
END;
/
2.4 替代触发器
语法同建立表触发器,使用关键词INTEAD_OF代替BEFORE或AFTER。
替代触发器解决了不能修改视图的问题,它只能在视图上创建,执行一个PL/SQL代码块,而不是一条DML语句
(如INSERT、UPDATE)。
编写替代触发器时,主要是编写更新视图底层的表的代码。同时也可以针对视图中的某个字段或所有字段。语
法:
CREATE OR REPLACE TRIGGER tri_emp_view
INSTEAD OF UPDATE ON emp_view
BEGIN
PL/SQL code;
END;
2.5 触发器的局限性
1. 触发器无法完成的工作:
1) 查询或修改变化的表;
2) 执行数据定义语言(CREATE、DROP、ALTER)的语句;
3) 执行COMMIT、ROLLBACK、SAVEPOINT 语句;
注:变化的表是指执行触发器时正在被修改的表。
触发器中不能执行COMMIT、ROLLBACK、SAVEPOINT 语句,以确保对事务控制的有效性及数据的完整性。
2. 如果使用触发器来实现商务规则或引用完整性规则,不会影响在触发器创建之前的记录。创建声明性约束实
际上是指出数据必须为真,如果出现了违反了约束的数据,则不能建立约束。
3. 不能对特定模式对象(例如:表)来定义DDL触发器。
 来自Oracle PL/SQL高级编程 作者对Oracle开发管理有多年的经验,并在Oracle 数据库的基础上开发了自己的交易控制中间件,适用于
融、电信、交通等多个行业,现就主要开发资料参考资料共享给大家。
祝大家在Oracle平台上更上一层楼,共同进步。
Categories: PL/SQL Tags:

ORACLE中的游标Cursor总结

March 6th, 2009 No comments

游标(Cursor):用来查询数据库,获取记录集合(结果集)的指针,可以让开发者一次访问一行结果集,在每条结果集上作操作。

 游标可分为:

1.       静态游标:分为显式(explicit)游标和隐式(implicit)游标。

2.       REF游标:是一种引用类型,类似于指针。

 

1、静态游标

1.1显式游标

定义格式:   

CURSOR 游标名 ( 参数 )  IS

Select 语句 FOR UPDATE [OF [schema.]table.column[,[schema.]table.column]..

[nowait]

例子1 :无参数,打开关闭游标

set serveroutput on size 10000000 ;

 

 create or replace procedure TEST is
  cursor c1 is
    select tname from tab;
  pname varchar2(
32);
begin
  open c1;
  loop
    fetch c1
      into pname;
    exit when c1%notfound;
    dbms_output.put_line(pname);
  end loop;
  close c1;

end TEST

exec test;

例子2 :参数使用,参数使用方法和存储过程一样

create or replace procedure TEST is
  cursor c1(pname in varchar2) is
    select tname from tab where tname like pname;
  pname varchar2(
32);
begin
  open c1(
‘T%’);
  loop
    fetch c1
      into pname;
    exit when c1%notfound;
    dbms_output.put_line(pname);
  end loop;
  close c1;

end TEST;

1.2隐式游标

不用明确建立游标变量,分两种:

1.在PL/SQL中使用DML语言,使用ORACLE提供的名为“SQL”的隐示游标。

举例:

declare
begin
  update departments set department_name = department_name;
  –where 1=2;

  dbms_output.put_line(
‘update ‘ || sql%rowcount || ‘ records’);
end;
/

2CURSOR FOR LOOP,用于for loop 语句

举例:

例子1:无参数,使用循环,无须打开关闭,本人这种方式

create or replace procedure TEST is
  cursor c1 is
    select tname from tab;
begin
  for rr in c1 loop
    dbms_output.put_line(rr.tname);
  end loop;

end TEST;

例子1:有参数,使用循环,无须打开关闭,本人这种方式

 

create or replace procedure TEST is
  cursor c1(pname in varchar2) is
    select tname from tab where tname like pname;
begin
  for rr in c1(
‘T%’) loop
    dbms_output.put_line(rr.tname);
  end loop;

end TEST;

 

 

1.3游标常用属性:

%FOUND:变量最后从游标中获取记录的时候,在结果集中找到了记录。

%NOTFOUND:变量最后从游标中获取记录的时候,在结果集中没有找到记录。

%ROWCOUNT:当前时刻已经从游标中获取的记录数量。

%ISOPEN:是否打开。

 

Declare  /* /* 定义静态游标 */ */
  Cursor emps is
    Select * from employees where rownum <
6 order by 1;

  Emp employees%rowtype;
  Row number := 1;
Begin
  Open emps; /* ´
打开静态游标 */
  Fetch emps
    into emp; /*
 读取游标当前行  */

  Loop
    If emps%found then
      Dbms_output.put_line(‘Looping over record ‘ || row || ‘ of ‘ ||
                           emps%rowcount);
      Fetch emps
        into emp;
      Row := row +
1;
    Elsif emps%notfound then
      Exit;
    End if;
  End loop;

  If emps%isopen then
    Close emps; /*  关闭游标  */
  End if;
End;
/

 

 1.4 游标的更新和删除

在PL/SQL中依然可以使用UPDATE和DELETE语句更新或删除数据行。显式游标只有在需要获得多行数据的情况下使用。PL/SQL提供了仅仅使用游标就可以执行删除或更新记录的方法。

 

 

UPDATE或DELETE语句中的WHERE CURRENT OF子串专门处理要执行UPDATE或DELETE操作的表中取出的最近的数据。要使用这个方法,在声明游标时必须使用FOR UPDATE子串,当对话使用FOR UPDATE子串打开一个游标时,所有返回集中的数据行都将处于行级(ROW-LEVEL)独占式锁定,其他对象只能查询这些数据行,不能进行UPDATE、DELETE或SELECT…FOR UPDATE操作。

在多表查询中,使用OF子句来锁定特定的表,如果忽略了OF子句,那么所有表中选择的数据行都将被锁定。如果这些数据行已经被其他会话锁定,那么正常情况下ORACLE将等待,直到数据行解锁。

在UPDATE和DELETE中使用WHERE CURRENT OF子串的语法如下:

WHERE{CURRENT OF cursor_name|search_condition}

 

 

 

create or replace procedure pc_SetVersionValid(PFlowsID in integer) is
  Cursor c1 is
    select *
      from wf_flows
     where flowname in
           (select flowname from wf_flows where flowsid = PFlowsID)
       for update;

  r c1%rowtype;
  v integer;
begin
  open c1;
  fetch c1
    into r;
  while c1%found loop
    if r.flowsid = PFlowsID then
      v := 1;
    else
      v :=
0;
    end if;
 
    UPDATE wf_flows SET isValid = v WHERE CURRENT OF c1;
 
    fetch c1
      into r;
 
  end loop;
  close c1;
  commit;
end;

 显式和隐式游标的区别:

尽量使用隐式游标,避免编写附加的游标控制代码(声明,打开,获取,关闭),也不需要声明变量来保存从游标中获取的数据。

 

2REF CURSOR游标

动态游标,在运行的时候才能确定游标使用的查询。可以分为:

create or replace procedure TEST is
  sqlstr varchar2(
500);
  type RefCur is ref cursor;
  c1 refcur;
begin
  sqlstr :=
‘select * from tab’;
  open c1 for sqlstr;
  close c1;
end;

 

 

REF CURSOR实现BULK功能

1. 可以加速INSERT, UPDATE, DELETE语句的执行,也就是用FORALL语句来替代循环语句。

2. 加速SELECT,用BULK COLLECT INTO 来替代INTO

 

SQL> create table tab2  as select empno ID, ename NAME, sal SALARY from emp where 1=2;

create or replace procedure REF_BULK is

/*  定义复杂类型 */

type empcurtyp  is ref cursor;

type idlist  is table of emp.empno%type;

type namelist  is table of emp.ename%type;

type sallist  is table of emp.sal%type;

  /* 定义变量  */

emp_cv  empcurtyp;

ids  idlist;

names namelist;

sals sallist;

row_cnt number;

begin

open emp_cv for select empno, ename, sal from emp;

fetch emp_cv  BULK COLLECT  INTO ids, names, sals;

将字段成批放入变量中,此时变量是一个集合

close emp_cv;

 

for i in ids.first .. ids.last loop

dbms_output.put_line(‘ || ids(i) || ‘ || names(i) ||’ salary=’ || sals(i));

end loop;

 

FORALL  i  IN  ids.first .. ids.last

insert into tab2 values (ids(i), names(i), sals(i));

commit;

select count(*) into row_cnt from tab2;

dbms_output.put_line(‘———————————–’);

dbms_output.put_line(‘The row number of tab2 is ‘ || row_cnt);

end REF_BULK;

 

 

 

 

3cursor ref cursor的区别

从技术底层看,两者是相同的。普通plsql cursor在定义时是“静态”的。而

Ref cursors可以动态打开。

例如下面例子:

Declare

type rc is ref cursor;

cursor c is select * from dual;

 

l_cursor rc;

begin

if ( to_char(sysdate,’dd’) = 30 ) then

       open l_cursor for ‘select * from emp’;

elsif ( to_char(sysdate,’dd’) = 29 ) then

       open l_cursor for select * from dept;

else

       open l_cursor for select * from dual;

end if;

open c;

end;

/

rc根据逻辑动态打开;而游标c定义好了只有就无法修改了。

ref cursor可以返回给客户端,cursor则不行。

cursor可以是全局的global ref cursor则必须定义在过程或函数中。

ref cursor可以在子程序间传递,cursor则不行。

cursor中定义的静态sqlref cursor效率高,所以ref cursor通常用在:向客户端返回结果集

 

Categories: oracle, PL/SQL Tags: ,

Oracle 10G 新特性——增强的CONNECT BY子句

March 6th, 2009 No comments
Categories: 10g新特性, oracle, PL/SQL Tags:

oracle 使用存储过程分割字符串为数组

March 6th, 2009 No comments

type tt_type is table of varchar2(32) INDEX BY BINARY_INTEGER;  

 –使用”,”将分割字符串转换为数组,返回为数组个数,

  function toArray(Liststr in varchar2, V1 out tt_type) return integer is
   

 

–FieldNames转化为数组


    TmpStr varchar2(
32);
    Str    varchar2(
4000);
    j      integer;
  begin
    Str := Liststr;
    j   :=
0;
    IF Instr(Liststr,
‘,’, 1, 1) = 0 THEN
      V1(j) := Liststr;
      j := j +
1;
    else
      While Instr(str,
‘,’, 1, 1) > 0 Loop
        TmpStr := Substr(str,
1, Instr(str, ‘,’, 1, 1) – 1);
     
        V1(j) := TmpStr;
        str := SubStr(Str, Instr(str,
‘,’, 1, 1) + 1, length(str));
        j := j +
1;
      end loop;
      if not str is null then
       –将最后一个保存

        V1(j) := str;
        j := j +
1;
      end if;
    end if;
    return J;
  end;

Categories: oracle, PL/SQL Tags:

oracle 使用存储过程分割字符串为数组

February 23rd, 2009 No comments

type tt_type is table of varchar2(32) INDEX BY BINARY_INTEGER;  

 使用”,”将分割字符串转换为数组,返回为数组个数,

  function toArray(Liststr in varchar2, V1 out tt_type) return integer is
   

 

–FieldNames转化为数组
    TmpStr varchar2(
32);
    Str    varchar2(
4000);
    j      integer;
  begin
    Str := Liststr;
    j   :=
0;
    IF Instr(Liststr,
‘,’, 1, 1) = 0 THEN
      V1(j) := Liststr;
      j := j +
1;
    else
      While Instr(str,
‘,’, 1, 1) > 0 Loop
        TmpStr := Substr(str,
1, Instr(str, ‘,’, 1, 1) – 1);
     
        V1(j) := TmpStr;
        str := SubStr(Str, Instr(str,
‘,’, 1, 1) + 1, length(str));
        j := j +
1;
      end loop;
      if not str is null then
     
 将最后一个保存
        V1(j) := str;
        j := j +
1;
      end if;
    end if;
    return J;
  end;

Categories: oracle, PL/SQL Tags:

轻松掌握如何从sqlplus向shell传递变量

February 14th, 2009 No comments

出处: http://zjc4u.itpub.net/post/4075/453468

很多人在实际的工作和学习中,都遇到了无法通过sqlplus向shell传递变量,下面我们来介绍一个实例脚本来进行折中处理。

[oracle@jumper oracle]$ cat a.sh
sqlplus -S “/ as sysdba” << !
set heading off
col today noprint
column today new_val dat
select to_char( sysdate, ”yyyy-mm-dd”) today from dual;
host echo ”today is ” &dat
exit;
exit;
!
[oracle@jumper oracle]$ ./a.sh

today is 2005-04-11

[oracle@jumper oracle]$

 

 

 

下面我们再来介绍另一个实例方法:

[oracle@jumper oracle]$ more a.sh

#!/bin/ksh
VALUE=`sqlplus -silent “/ as sysdba” < < END
set pagesize 0 feedback off verify off heading off echo off
select max(sequence#) from v$log_history;
exit;
END`

if [ -z "$VALUE" ]; then
echo “No rows returned from database”
exit 0
else
echo “Max Sequence Number: $VALUE”
fi

[oracle@jumper oracle]$ ./a.sh
Max Sequence Number: 17

Categories: oracle, PL/SQL Tags: ,

PL/SQL存储过程报 trailing null missing from STR bind value 的问题

February 2nd, 2009 No comments

我们在写PL/SQL存储过程定义字符型参数类型时一般都是varchar2,做为输入参数超过1333个英文字符串(测试得:1回车相当于3英文,1汉字相当于2英文)时就报trailing null missing from STR bind value。

解决此问题我们可以定义参数类型为long型,我们数据库表结构定义为varchar2(4000)时输入少于4000的字符串就不会再报这错。

Categories: PL/SQL Tags:

Videos, Slideshows and Podcasts by Cincopa Wordpress Plugin