0

oracle如何在一个存储过程中调用另一个返回游标的存储过程

by 刘武 24. 六月 2009 21:24

实际项目当中经常需要在一个存储过程中调用另一个存储过程返回的游标,本文列举了两种情况讲述具体的操作方法。

第一种情况是返回的游标是某个具体的表或视图的数据,如:

CREATE OR REPLACE PROCEDURE P_TESTA (
PRESULT OUT SYS_REFCURSOR
)
AS
BEGIN
 OPEN PRESULT FOR SELECTFROM USERS;
END P_TESTA;
其中USERS就是数据库中一个表。在调用的时候只要声明一个该表的ROWTYPE类型就可以了:

CREATE OR REPLACE PROCEDURE P_TESTB
AS
VARCURSOR SYS_REFCURSOR;
R USERS%ROWTYPE;
BEGIN
      P_TESTA(VARCURSOR);
      LOOP
       FETCH VARCURSOR INTO R;
       EXIT WHEN VARCURSOR%NOTFOUND;
       DBMS_OUTPUT.PUT_LINE(R.NAME);
      END LOOP;
 END P_TESTB;

第二种情况,我们返回的不是表的所有的列,或许只是其中一列或两列,如:

CREATE OR REPLACE PROCEDURE P_TESTA (
PRESULT OUT SYS_REFCURSOR
)
AS
BEGIN
 OPEN PRESULT FOR SELECT ID,NAME FROM USERS;
END P_TESTA;
这里我们只返回了USERS表的ID,NAME这两个列,那么调用的时候也必须做相应的修改:

CREATE OR REPLACE PROCEDURE P_TESTB
AS
VARCURSOR SYS_REFCURSOR;
CURSOR TMPCURSOR IS SELECT ID,NAME FROM USERS WHERE ROWNUM=1;
R TMPCURSOR%ROWTYPE;
BEGIN
 P_TESTA(VARCURSOR);
    LOOP
     FETCH VARCURSOR INTO R;
        EXIT WHEN VARCURSOR%NOTFOUND;
        DBMS_OUTPUT.PUT_LINE(R.ID);
    END LOOP;
 END P_TESTB;
与之前不同的是我们声明了一个游标类型的变量TMPCURSOR ,注意TMPCURSOR 的结构必须与存储过程P_TESTA 返回的游标结构一致,否则就会出现错误。同理只要保持两个游标类型结构一致,就可以实现自由调用。

Tags: , ,

技术生涯

4

ora-04091 表XX发生了变化 触发器/函数不能读 的错误处理

by 刘武 11. 五月 2009 21:26

开发过程中碰到了ora-04091错误:表XX发生了错误,触发器/函数不能读

以下是模拟场景:

表PRODUCTS:

CREATE TABLE CKSP.PRODUCTS
(
    ID     NUMBER(1)     NOT NULL,
    NAME   VARCHAR2(100) NOT NULL,
    PRICE1 NUMBER(1)     NOT NULL,
    PRICE2 NUMBER(1)     NOT NULL
)
TABLESPACE USERS
NOLOGGING
PCTFREE 10
PCTUSED 0
INITRANS 1
MAXTRANS 255
STORAGE(BUFFER_POOL DEFAULT)
NOPARALLEL
NOCACHE

 

并声明触发器TRG_PRODUCTS_INSERT:

CREATE OR REPLACE TRIGGER TRG_PRODUCTS_INSERT
BEFORE INSERT
ON PRODUCTS
REFERENCING OLD AS OLD NEW AS NEW
FOR EACH ROW
declare 
varCount number;
begin  
	select count(id) into varCount from products where name = :new.name;
	if varCount > 0 then
		varCount:= varCount +1;
		:new.name :=varCount||'_'||:new.name;
	end if;
    
end;

 

若执行以下SQL则会引发ORA-04091错误

begin
    insert into products values(1,'p1',1,2);
    insert into products 
    select 2 as id ,name ,price1,price2 from products where id= 1;
end;

 

解决方法:修改SQL 如下

declare
    varName varchar2(100);
    varPrice1 number;
    varPrice2 number;
begin
    insert into products values(1,'p1',1,2);
    select name,price1,price2 into varName,varPrice1,varPrice2 from products where id =1;
    insert into products values(2,varName,varprice1,varPrice2);    
end;

 

在此情形中不能使用INSERT INTO XXX SELECT .... 语句,而应使用变量来保存值

这也许只是众多情形之一,仅供参考.

Tags: ,

技术生涯

0

常用的ORACLE PL/SQL管理命令一

by 刘武 4. 五月 2009 21:11

熟悉ORACLE管理的一定对这些命令不会陌生,不过对于我这个刚接触ORACLE管理的来说,还是有必要做下记录,以便随时查看。

一 登录SQLPLUS

sqlplus 用户名/密码@数据库实例 as 登录角色;

如:用户sys(密码为123)以sysdba的角色登录数据库ORACL,我们可以输入:sqlplus sys/123@oracl as sysdba;

这种登录方式会直接暴露密码,如果想隐藏密码,可以在此省略密码的输入,如:sqlplus sys@oracl as sysdba;回车以后ORACLE会给出输入密码的提示符。

登录以后如果想切换其他的用户,可以直接使用connect 命令,如:connect user2/password@oracl as sysdba,同上一样,可以将密码分开输入。

二 退出SQLPLUS

quit;

三 创建用户

create user 用户名 identified by 密码;

如:创建用户CKSP,密码为123: create user cksp identified by 123;

四 给用户分配角色或权限

grant *** to ***;

如:给刚才的用户分配角色DBA:grant dba to cksp;

分配create table 权限:grant

五 删除用户

drop user 用户 [cascade];

其中cascade是可选的,如果输入了,则表示删除该用户及所有数据。

如:删除上面创建的用户CKSP及他的所有数据:drop user cksp cascade;

Tags: ,

技术生涯

0

Oracle 如何对中文字段进行排序

by 刘武 10. 二月 2009 22:17

Oracle 中对中文字段进行排序通常有三种方式

1)按笔画排序

 select * from Table order by nlssort(columnName,'NLS_SORT=SCHINESE_STROKE_M')

2)按部首排序

select * from Table order by nlssort(columnName,'NLS_SORT=SCHINESE_RADICAL_M')

3)按拼音排序

select * from Table order by nlssort(columnName,'NLS_SORT=SCHINESE_PINYIN_M');

Tags:

技术生涯

Powered by BlogEngine.NET 1.6.1.9  登录
Original Design by Laptop Geek, Adapted by onesoft