1. 苏葳的备忘录首页
  2. 数据库

Oracle的PLS-00231 错误分析

oracle package functionSybase中没有Function的概念(Java的不算),所以存储过程都需要用Exec调用。而Oracle这方面更灵活一些,可以声明Function和Procedure。除了语法略有差异,Function必须作为表达式的一部份调用,且可以用return返回一个值。而且Oracle更提供了Package语法,可以将一组相关PL/SQL元素封装起来,提供一些面向对象特点,类似于其它语言中的类,当然,它不需要实例化。而在内部,Sql和PL/SQL似乎是两套相对独立的解析器。

create or replace function Two
      return Number
      is
      begin
        return 2;
      end Two;
create or replace  procedure PrintTwo
      is
       myNum Number;
     begin
      select Two()
      into myNum
      from dual;
      dbms_output.put_line(myNum);
    end PrintTwo;

以上函数和过程,放在函数和过程中声明,调用正常。注意two函数被printtwo过程以sql语句形式调用。

若将此二程序段放入一个package中,函数私有方式,则包体编译时出错,报pls-00231错误,函数不能在sql语句中使用之类。

原因为何?

pl sql和sql对于oracel而言,是两个不同的引擎,因而对于sql引擎而言,plsql包中私有的two函数,是无法发现的。

所以包中使用two函数,方法有两种:

1 在包头声明中声明two函数,即将two函数作公有声明,此时可以select two into mynum from dual;形式调用,当然这样将无法隐藏two函数。

2 以mynum:=two;方式调用,由于此种方式以pl sql引擎解析,所以可以正常使用以私有方式声明的two函数。

可见需以合适的方式将two函数暴露给sql或plsql引擎方可使用。那么若不将two作公有声明,而用包名.two方式调用,不一样可定位到two函数?哦当然不行,你忘了two未作公有声明,在包外无论如何都是不可见的哈哈。

那么对于包内一个返回集合类型的函数,如:

FUNCTION strsplit(p_list IN VARCHAR2, p_sep IN VARCHAR2 DEFAULT '|')
    RETURN str_split
    PIPELINED;

str_split为一个自定义集合类型:

TYPE "STR_SPLIT" IS TABLE OF VARCHAR2 (4000);

此时,无论将type str_split以公有,或私有方式 在包中声明,均无法正常使用sql方式调用,由于返回为集合,也不能用plsql的kkk:=str_split方式调用。此时,只能将str_split定义为oracle的自定义数据类型:

create or replace TYPE "STR_SPLIT" IS TABLE OF VARCHAR2 (4000)

然后去除包内公有或私有的type str_split定义,即可使用sql方式调用函数。

原创文章,作者:苏葳,如需转载,请注明出处:https://www.swmemo.com/361.html

发表评论

邮箱地址不会被公开。 必填项已用*标注