什么是子程序:命名的 PL/SQL 块,编译并存储在数据库中
1.存储过程
1.1 语法结构
sql
CREATE [OR REPLACE] PROCEDURE
<procedure name> [(<parameter list>)]
IS|AS
<local variable declaration>
BEGIN
<executable statements>
[EXCEPTION
<exception handlers>]
END;
1.2 案例讲解


参数的三种类型
-
in 输入
-
out 输出
-
in out 输入输出
案例讲解
sql
create or replace procedure protest03(
p_name in varchar2,
p_sex in out varchar2,
p_age in out number)
is
-- 声明变量
begin
dbms_output.put_line(p_name || ',' || p_sex || ',' || p_age );
p_sex :='我是P_sex';
end ;
调用过程
sql
declare
v_name varchar2(10) :='&请输入名字' ;
v_sex varchar2(50) :='&请输入性别';
v_age number(3) :='&请输入年龄';
begin
protest03(v_name, v_sex , v_age); -- 当没有参数里,括号可省略不写
dbms_output.put_line(v_sex);
end;
案例: 请根据性别或名字查询相关记录,并把结果 返回来 打印了出来 提示用 sys_refcursor
sql
create or replace procedure protest04(
p_name varchar2,
p_sex varchar2,
myresult out sys_refcursor)
is
v_sql varchar2(100) ;
begin
v_sql :='select * from t_student where 1=1 ';
if p_name is not null then
v_sql :=v_sql || ' and stuname like ''%' || p_name || '%'' ' ;
end if;
if p_sex is not null then
v_sql := v_sql || ' and sex = ''' || p_sex || ''' ';
end if;
dbms_output.put_line('v_sql=' || v_sql);
open myresult for v_sql ;
end;
调用
sql
-- 执行测试
declare
v_name varchar2(20) :='&请输入名字';
v_sex varchar2(4) :='&请输入性别' ;
mycursor sys_refcursor ;
v_row t_student%rowtype;
begin
-- protest04(v_name, v_sex , mycursor);
-- 1) 位置传递
--2 )名称传递
--protest04( p_sex =>v_sex , p_name => v_name , myresult => mycursor);
--3) 混合使用 : 先用位置传递,如果后面有用了名称传递,后面就不能用位置传递
protest04(v_name , myresult=>mycursor ,p_sex => v_sex,);
loop
fetch mycursor into v_row ;
exit when mycursor%notfound ;
dbms_output.put_line(v_row.stuname || ',' || v_row.sex || ',' || v_row.age );
end loop;
close mycursor ;
end;
2.存储函数
类似于java中方法,有返回值可以返回值的命名的 PL/SQL 子程序。
2.1 语法结构
sql
CREATE [OR REPLACE] FUNCTION
<function name> [(param1,param2)]
RETURN <datatype> IS|AS
[local declarations]
BEGIN
Executable Statements;
RETURN result;
EXCEPTION
Exception handlers;
END;
2.2 案例讲解
无参案例:写一个函数 ,获取学生名称
sql
create or replace function funtest02
return varchar2
is
v_name varchar2(20) ;
begin
select stuname into v_name from t_student where id=201;
return v_name;
end ;
如何调用呢
-
a)用PL/SQL块调用函数
-
b)用select语句调用: 在函数中不能有增加删除修改的语句,只能是查询的语句
有参案例:

3.程序包
程序包:作用就是管理我们的存储过程和方法
sql
-- 程序包:作用就是管理我们的存储过程和方法
-- 规范和主体两部分组成
-- 创建一个规范
create or replace package pak01
is
procedure myprocdure01(p_name varchar2);
function myfun01 return number;
end pak01;
-- 创建规范对应的主体:主体中的方法如果在规范中声明了。那么外包可以访问。如果没有那么就只能被内部调用
create or replace package body pak01
is
-- myprodure01 存储过程
procedure myprocdure01(p_name varchar2)
as
begin
dbms_output.put_line('p_name = '||p_name);
end;
-- myfun01 方法
function myfun01 return number
is
begin
dbms_output.put_line('方法执行了.... ');
return 666;
end;
end pak01;
create or replace package body pak01
is
-- myprodure01 存储过程
procedure myprocdure01(p_name varchar2)
as
begin
dbms_output.put_line('p_name = '||p_name);
end;
-- myfun01 方法
function myfun01 return number
is
begin
dbms_output.put_line('方法执行了.... ');
return 666;
end;
function myfun02 return number
is
begin
dbms_output.put_line('方法执行了..222.. ');
return 999;
end;
end pak01;
-- 调用package中的过程和方法 package.
begin
pak01.myprocdure01('李四');
end;
select pak01.myfun02 from dual;