日期:2014-05-16  浏览次数:20445 次

oracle 存储过程和函数
--子程序
/*
  子程序是被命名的PL/SQL块,可以带参数,多次被调用,模块化
  过程&&函数
  过程:执行特定操作
  函数:返回特定数据
  
  定义:
  CREATE [OR REPLACE] PROCEDURE procedure_name(argument [mode] datatype...)
  IS|AS
  --这里可以声明一些变量,相当于declare块,不过没有declare关键字
  BEGIN
     statement...
  END;
  -------******-------
      procedure_name 过程名
      argument 参数名
      mode 参数类型, IN 输入参数 || OUT 输出参数 ||IN OUT . 默认是输入参数
      datatype 参数类型 ,不需要指定长度
    
*/
--不带参数
CREATE OR REPLACE PROCEDURE p_time
IS
BEGIN
   dbms_output.put_line(sysdate);
END;
--调用
call p_time();


--默认IN型参数
CREATE OR REPLACE PROCEDURE emp_1(num int)
AS
   v_name emp.ename%type;
   v_deptno emp.deptno%type;
BEGIN
   IF num<10000 THEN--根据传入的参数,是否执行查询
      select ename,deptno into v_name,v_deptno from emp where emp.empno=num;
      dbms_output.put_line(v_name||'--'||v_deptno);
   ELSE 
      dbms_output.put_line('too big');
   END IF;
END;
--调用
BEGIN
   emp_1(77880);
END;


--明确定义参数模式
/*
   IN      read-only :pass values into 相当于一个常量
   OUT     write-only :pass values back 相当于一个变量
   IN OUT  read/write
*/
CREATE OR REPLACE PROCEDURE emp_count_1(v_deptid IN int,v_count OUT int)--一个IN,一个OUT
IS
BEGIN
  select count(emp.empno) into v_count from emp where emp.deptno=v_deptid;
  dbms_output.put_line(v_count);
END;
--调用
DECLARE 
   v_count int;
BEGIN
   --传递OUT类型的参数是传递一个类型相符的变量,因为out的类型的是要作为输出的,所以需要具体的变量来存放
   emp_count_1(10,v_count);
   --传递参数默认按定义顺序,但也可以如下按照参数名称
   emp_count_1(v_count=>v_count,v_deptid => 10);
   --当然也可以两种方式混合使用
   emp_count_1(10,v_count => v_count);
END;

--感觉out的参数很麻烦的,还不如在过程内直接定义一个变量
CREATE OR REPLACE PROCEDURE emp_count_2(v_deptid IN INT)
IS
   v_count int;
BEGIN
   select count(emp.empno) into v_count from emp where emp.deptno=v_deptid;
   dbms_output.put_line(v_count);
END;
--调用
BEGIN
   emp_count_2(10);
END;

--过程调用过程
CREATE OR REPLACE PROCEDURE p_main
IS
BEGIN--调用上面定义的两个过程
   p_time();
   emp_count_2(10);
END;  
--调用
BEGIN
   p_main();
END;



--函数
/*
 有返回值,其它的和过程差不多
 
 CREATE [OR REPLACE] FUNCTION function_name(argument [mode] datatype...)
 RETURN datatype --返回类型必须
 IS|AS
 [variable datatype...]
 BEGIN
     statement...
     Pl/SQL块中必须要有一条return语句
 END;
*/

--
CREATE OR REPLACE FUNCTION get_name(u_id IN int) RETURN varchar2
AS
   v_name emp.ename%type;
BEGIN
   select ename into v_name from emp where emp.empno=u_id;
   RETURN v_name;
END;
--调用
DECLARE
   v_name emp.ename%type;
   v_job emp.job%type;
BEGIN
   v_name :=get_name(7788);--直接给变量赋值
   select job into v_job from emp where ename=get_name(7788) and rownum<2;--在查询语句中调用
   dbms_output.put_line(v_name);
END;


--返回record
CREATE OR REPLACE FUNCTION get_record(pno in number) RETURN emp%rowtype
IS
   v_record emp%rowtype;
BEGIN
  select *  into v_record from emp where empno=pno;
  return v_record;
END;

--
DECLARE 
   v_record emp%rowtype;
BEGIN
   v_record := get_record(7788);
   dbms_output.put_line(v_record.ename||'--'||v_record.job);
END;

--如果一个函数没有参数,那么调用他的时候可以不用加()


--exp 
CREATE OR REPLACE FUNCTION get_num(v_id in int) RETURN INT
IS
 CURSOR v_cs is select count(empno) tt,max(empno) mx from emp where emp.deptno=v_id group by deptno;
 v_num v_cs%rowtype;
BEGIN
 OPEN v_cs;
 FETCH v_cs into v_num;
 dbms_output.put_line(v_num.tt||'--'||v_num.mx);
 close v_cs;
 RETURN v_num.tt;
END;
--
select get_num(10) from dual;

--综合例子
--统计emp表中每个部门的人员
DECLARE
  --声明一个嵌套表用来存放要查询的部门编号
  TYPE v_array IS VARRAY(5) OF INT;
  v_arr v_array;
  --声明一个嵌套表用来存放每一个部门的员工名字
  TYPE v_list_type IS TABLE OF VARCHAR2(30);
  v_list v_list_type := v_list_type();
  --声明一个函数 根据部门查询统计该部门的员工总数和员工名字
  FUNCTION get_num(v_dpno IN INT) RETURN v_list_type
  IS
    v_num INT;
    v_i int :=1;
    v_name varchar2(30);
    v_names v_list_type := v_list_type();
    TYPE v_cs_type IS REF CURSOR;
    v_cs v_cs_type;
    
  BEGIN
    --统计员工总数
    select count(empno) into v_num from emp where emp.deptno= v_dpno;
    dbms_output.put_line('编号是:'||v_dpno||'的部门的员工数是:'||v_num);
    --查询该部门员工名字
    OPEN v_cs FOR select