ORACLE数据库PL/SQL编程之把过程与函数说透

数据库 Oracle
过程与函数(另外还有包与触发器)是命名的PL/SQL块(也是用户的方案对象),被编译后存储在数据库中,以备执行。因此,其它PL/SQL块可以按名称来使用他们。本文主要就PL/SQL中过程和函数的调用进行了说明,希望能对读者有所帮助。

过程函数统称为PL/SQL子程序,他们是被命名的PL/SQL块,均存储在数据库中,并通过输入、输出参数或输入/输出参数与其调用者交换信息。过程和函数的唯一区别是函数总向调用者返回数据,而过程则不返回数据。在本文中,主要介绍:

1、创建存储过程和函数。

2、正确使用系统级的异常处理和用户定义的异常处理。

3、建立和管理存储过程和函数。

创建函数

1. 创建函数

语法如下:

  1. CREATE [OR REPLACE] FUNCTION function_name  
  2.  
  3. (arg1 [ { IN | OUT | IN OUT }] type1  
  4.  
  5. [DEFAULT value1],  
  6.  
  7. [arg2 [ { IN | OUT | IN OUT }] type2 [DEFAULT value1]],  
  8.  
  9. ......  
  10.  
  11. [argn [ { IN | OUT | IN OUT }] typen [DEFAULT valuen]])  
  12.  
  13. [ AUTHID DEFINER | CURRENT_USER ]RETURN return_type  
  14.  
  15. IS | AS  
  16.  
  17. <类型.变量的声明部分> BEGIN 

执行部分

RETURN expressionEXCEPTION

异常处理部分END function_name;

IN,OUT,IN OUT是形参的模式。若省略,则为IN模式。IN模式的形参只能将实参传递给形参,进入函数内部,但只能读不能写,函数返回时实参的值不变。OUT模式的形参会忽略调用时的实参值(或说该形参的初始值总是NULL),但在函数内部可以被读或写,函数返回时形参的值会赋予给实参。IN OUT具有前两种模式的特性,即调用时,实参的值总是传递给形参,结束时,形参的值传递给实参。调用时,对于IN模式的实参可以是常量或变量,但对于OUT和IN OUT模式的实参必须是变量。

一般,只有在确认function_name函数是新函数或是要更新的函数时,才使用OR REPALCE关键字,否则容易删除有用的函数。

#p#

例1、获取某部门的工资总和:

--获取某部门的工资总和

  1. CREATE OR REPLACEFUNCTION get_salary(  
  2.  
  3. Dept_no NUMBER,  
  4.  
  5. Emp_count OUT NUMBER)  
  6.  
  7. RETURN NUMBER IS  V_sum NUMBER;BEGIN  SELECT SUM(SALARY), count(*) INTO V_sum, emp_count  
  8.  
  9. FROM EMPLOYEES WHERE DEPARTMENT_ID=dept_no;  
  10.  
  11. RETURN v_sum;EXCEPTION  
  12.  
  13. WHEN NO_DATA_FOUND THEN  
  14.  
  15. DBMS_OUTPUT.PUT_LINE('你需要的数据不存在!');  
  16.  
  17. WHEN OTHERS THEN  
  18.  
  19. DBMS_OUTPUT.PUT_LINE(SQLCODE||'---'||SQLERRM);END get_salary; 

2. 函数的调用

函数声明时所定义的参数称为形式参数,应用程序调用时为函数传递的参数称为实际参数。应用程序在调用函数时,可以使用以下三种方法向函数传递参数:

第一种参数传递格式:位置表示法。

即在调用时按形参的排列顺序,依次写出实参的名称,而将形参与实参关联起来进行传递。用这种方法进行调用,形参与实参的名称是相互独立,没有关系,强调次序才是重要的。

格式为:argument_value1[,argument_value2 …]

例2:计算某部门的工资总和:

  1. DECLARE  V_num NUMBER;  
  2.  
  3. V_sum NUMBER;BEGIN  
  4.  
  5. V_sum :=get_salary(10, v_num);  
  6.  
  7. DBMS_OUTPUT.PUT_LINE('部门号为:10的工资总和:'||v_sum||',人数为:'||v_num);END; 

第二种参数传递格式:名称表示法。

即在调用时按形参的名称与实参的名称,写出实参对应的形参,而将形参与实参关联起来进行传递。这种方法,形参与实参的名称是相互独立的,没有关系,名称的对应关系才是最重要的,次序并不重要。

格式为: argument => parameter [,…]

其中:argument 为形式参数,它必须与函数定义时所声明的形式参数名称相同parameter 为实际参数。

在这种格式中,形势参数与实际参数成对出现,相互间关系唯一确定,所以参数的顺序可以任意排列。

例3:计算某部门的工资总和:

  1. DECLARE  V_num NUMBER;  
  2.  
  3. V_sum NUMBER;BEGIN  
  4.  
  5. V_sum :=get_salary(emp_count => v_num, dept_no => 10);  
  6.  
  7. DBMS_OUTPUT.PUT_LINE('部门号为:10的工资总和:'||v_sum||',人数为:'||v_num);END; 

 #p#

第三种参数传递格式:组合传递。

即在调用一个函数时,同时使用位置表示法和名称表示法为函数传递参数。采用这种参数传递方法时,使用位置表示法所传递的参数必须放在名称表示法所传递的参数前面。也就是说,无论函数具有多少个参数,只要其中有一个参数使用名称表示法,其后所有的参数都必须使用名称表示法。

例4:

  1. CREATE OR REPLACE FUNCTION demo_fun(  
  2.  
  3. Name VARCHAR2,--注意VARCHAR2不能给精度,如:VARCHAR2(10),其它类似  
  4.  
  5. Age INTEGER,  
  6.  
  7. Sex VARCHAR2)  
  8.  
  9. RETURN VARCHAR2 AS  
  10.  
  11. V_var VARCHAR2(32);BEGIN  
  12.  
  13. V_var :name||':'||TO_CHAR(age)||'岁.'||sex;  RETURN v_var;END;DECLARE  
  14.  
  15. Var VARCHAR(32);BEGIN  Var :demo_fun('user1', 30, sex => '男');  
  16.  
  17. DBMS_OUTPUT.PUT_LINE(var);  Var :demo_fun('user2', age => 40, sex => '男');  
  18.  
  19. DBMS_OUTPUT.PUT_LINE(var);  Var :demo_fun('user3', sex => '女', age => 20); 

无论采用哪一种参数传递方法,实际参数和形式参数之间的数据传递只有两种方法:传址法和传值法。所谓传址法是指在调用函数时,将实际参数的地址指针传递给形式参数,使形式参数和实际参数指向内存中的同一区域,从而实现参数数据的传递。这种方法又称作参照法,即形式参数参照实际参数数据。输入参数均采用传址法传递数据。

传值法是指将实际参数的数据拷贝到形式参数,而不是传递实际参数的地址。默认时,输出参数和输入/输出参数均采用传值法。在函数调用时,ORACLE将实际参数数据拷贝到输入/输出参数,而当函数正常运行退出时,又将输出形式参数和输入/输出形式参数数据拷贝到实际参数变量中。

3. 参数默认值

在CREATE OR REPLACE FUNCTION 语句中声明函数参数时可以使用DEFAULT关键字为输入参数指定默认值。

例5:

  1. CREATE OR REPLACE FUNCTION demo_fun(  Name VARCHAR2,  
  2.  
  3. Age INTEGER,  
  4.  
  5. Sex VARCHAR2 DEFAULT '男')  
  6.  
  7. RETURN VARCHAR2 AS  
  8.  
  9. V_var VARCHAR2(32);BEGIN  
  10.  
  11. V_var :name||':'||TO_CHAR(age)||'岁.'||sex;  
  12.  
  13. RETURN v_var;END; 

具有默认值的函数创建后,在函数调用时,如果没有为具有默认值的参数提供实际参数值,函数将使用该参数的默认值。但当调用者为默认参数提供实际参数时,函数将使用实际参数值。在创建函数时,只能为输入参数设置默认值,而不能为输入/输出参数设置默认值。

  1. DECLARE var VARCHAR(32);BEGIN Var :demo_fun('user1', 30);  
  2.  
  3. DBMS_OUTPUT.PUT_LINE(var); Var :demo_fun('user2', age => 40);  
  4.  
  5. DBMS_OUTPUT.PUT_LINE(var); Var :demo_fun('user3', sex => '女',  
  6.  
  7. age => 20); DBMS_OUTPUT.PUT_LINE(var);END; 

关于PL/SQL编程中函数和过程的相关知识就介绍到这里,如果想了解更多Oracle数据库的知识请到我们网站的Oracle专栏:http://database.51cto.com/oracle/,谢谢大家的支持!

【编辑推荐】

  1. 因为Oracle推EF for Oracle引发的口水战
  2. Oracle数据库使用OMF来简化数据文件的管理
  3. 浅谈禁用以操作系统认证方式登录Oracle数据库
  4. Oracle数据导入MySQL的快捷工具:MySQL Migration Toolkit
  5. 浅析64位win7下使用PL/SQL Developer连接远程Oracle数据库
责任编辑:赵鹏 来源: 中国IT实验室
相关推荐

2010-05-05 11:17:55

Oracle数据库

2011-08-23 17:33:00

SQLAndroid

2010-04-12 08:59:00

2011-04-14 13:01:53

Oracle数据库

2011-08-29 13:24:50

Oracle数据库PLSQL设置快捷键

2009-07-24 10:29:29

PL SQL编程规范

2011-04-07 10:50:47

数据库编程注释规范

2009-03-09 16:39:19

PL SQLOracle过程调试

2011-05-26 14:07:11

SQL ServerOracle数据库镜像对比

2009-03-27 13:15:20

OracleSQL Server镜像

2011-04-07 10:29:21

数据库编程书写规范

2011-07-29 13:40:34

Oracle数据库PLSQL异常处理

2010-04-13 14:35:17

2009-03-26 10:46:58

OraclePL、SQL存储过程

2010-10-25 15:56:02

Oracle PLSQL过程调试

2010-04-16 13:53:23

Oracle数据库

2009-03-09 17:46:16

ASP.NETSQLOracle

2011-08-24 10:45:23

Oracle数据库进程从属进程

2010-04-14 10:12:07

Oracle数据库

2010-09-10 15:11:07

SQLGetKey函数
点赞
收藏

51CTO技术栈公众号