Oracle学习笔记(十二)

十三、存储过程和存储函数
1、掌握存储过程(相当于建立一个函数或者方法体,然后通过外部对其调用)
指存储在数据库中供所有程序调用的子程序叫做存储过程或存储函数。

相同点:
完成特定功能的程序
区别:
是否用return语句返回值

(1)创建和使用存储过程
用create procedure命令建立存储过程和存储函数

语法:
create or replace procedure 过程名(参数列表) as PL/SQL 子程序体(说明部分);

事例:
(a)打印一个存储过程:打印HelloWorld(无参数)
create or replace procedure sayhelloworld
as
--说明部分
begin
dbms_output.put_line(‘Hello World !!!‘);
end;
/
在编译器左边的过程里形成大写的SAYHELLOWORLD

对其调用:
方法一:
execute 简写为exec sayhelloworld();
方法二:
begin
sayhelloworld();
sayhelloworld();
end;
/

(b)带参数的存储过程
事例:为指定的员工,涨100块钱的工资,并且打印涨前和涨后的薪水
思路:1、创建一个带参数的存储过程
2、给指定的员工涨100块钱的工资,并且打印涨前和涨后的薪水
create or replace procedure raisesalary(eno in number)
as
--定义一个变量保存涨钱的薪水
psal emp.sal%type;
begin
--得到员工涨前的薪水
select sal into psal from emp where empno=eno;

--给员工涨100
update emp set sal=sal+100 where empno=eno;
dbms_output.put_line(‘涨前薪水 :‘||psal||‘涨后薪水 :‘||(psal+100));

--是否需要commit?
--注意:一般不在存储过程或者存储函数中,commit和rollback.原则不绝对
end;
/

如何调用:
begin
raisesalary(7839);
raisesalary(7566);
commit;
end;

如何调试
注意:不推荐远程调试,推荐本地调试
过程-->右键-->编译以进行调试

设置断点
select sal into psal from emp where empno=eno;
点击瓢虫调试
更改值 ENO := 7839; 点击确定
授权
系统认证登录:sqlplus / as sysdba
grant DEBUG CONNECT SESSION ,DEBUG ANY PROCEDURE to scott;
右键点击检测psal
调试
点击F8

2、掌握存储函数
函数为一命名的存储程序,可带参数,并返回一计算值.
函数和过程的结构类似,但必须有一个return子句,用于返回函数值.

(1)创建存储函数的语法
create or replace function 函数名(参数列表)
return 函数值类型
as
PL/SQL子程序体;

事例:查询某个员工的年收入
create or replace function queryempincome(eno in number)
return number
as
--定义变量保存员工的薪水和奖金
psal emp.sal%type;
pcomm emp.comm%type;
begin
--得到该员工的月薪和奖金
select sal,comm into psal,pcomm from emp where empno=eno;
--直接返回年收入
return psal*12+nvl(pcomm,0);
end;
/

set linesize 200
select * from emp;

调用存储函数

(2)in和out参数
一般来讲,存储过程和存储函数的区别在于存储函数可以有一个返回值,而存储过程没有返回值.

存储过程和存储函数都可以有out参数
存储过程和存储函数都可以有多个out参数
存储过程可以通过out参数来实现返回值

什么时候用存储过程/存储函数?
原则:
-- 如果只有一个返回值,用存储函数,否则,就用存储过程.

----out参数,查询某个员工姓名,月薪和职位
create or replace procedure queryempinform(eno in number, pename out varchar2,psal out number,pjob out varchar2)
as
begin
--得到该员工的姓名,月薪和职位
select ename,sal,empjob into pename,psal,pjob from emp where empno=eno;
end;
/

思考:(1)查询某个员工的所有信息---->out参数太多了
set linesize 80
desc emp
(2)查询某个部门中所有员工的所有信息--->out中返回集合

(3)在应用程序中访问存储过程和存储函数
(a)访问存储过程
public class JdbcUtils {
private static String url = "jdbc:oracle:thin:@localhost:1521:orcl";
private static String username = "scott";
private static String password = "tiger";
private static Connection conn;

public static Connection getConnection() {
conn = null;
try {
conn = DriverManager.getConnection(url, username, password);
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException("oracle连接获取失败");
}
return conn;
}

// 关闭连接对象
public static void close(Connection conn) {
if (conn != null) {
try {
conn.close();
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException("oracle连接关闭失败");
}
}
}

public static void close(Statement stmt) {
if (stmt != null) {
try {
stmt.close();
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException("oracle连接关闭失败");
}
}
}

public static void close(ResultSet rs) {
if (rs != null) {
try {
rs.close();
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException("oracle连接关闭失败");
}
}
}
}

public class TestProcedure {

@Test
public void testProcedure() {
String sql = "{call queryempinform(?,?,?,?)}";
Connection conn = null;
CallableStatement call = null;

try {
// 得到一个连接
conn = JdbcUtils.getConnection();
// 通过连接创建出statement
call = conn.prepareCall(sql);

//对于in参数,赋值
call.setInt(1, 7839);
//对于out参数,申明
call.registerOutParameter(2, OracleTypes.VARCHAR);
call.registerOutParameter(3, OracleTypes.NUMBER);
call.registerOutParameter(4, OracleTypes.VARCHAR);

//执行调用
call.execute();

//取出结果
String name =call.getString(2);
double sal =call.getDouble(3);
String job =call.getString(4);
System.out.println(name+"\t"+sal+"\t"+job);

} catch (Exception e) {
e.printStackTrace();
}finally{
JdbcUtils.close(conn);
JdbcUtils.close(call);
}
}

}
(b)访问存储函数
public class TestFunction {

@Test
public void testFunction() {
String sql = "{?=call queryempincome(?)}";
Connection conn = null;
CallableStatement call = null;

try {
// 得到一个连接
conn = JdbcUtils.getConnection();
// 通过连接创建出statement
call = conn.prepareCall(sql);

// 对于输出参数,申明
call.registerOutParameter(1, OracleTypes.NUMBER);

// 对于参数赋值
call.setInt(2, 7839);

// 执行调用
call.execute();

// 取出结果

double income = call.getDouble(1);

//打印结果集
System.out.println("年收入:" + income);

} catch (Exception e) {
e.printStackTrace();
} finally {
JdbcUtils.close(conn);
JdbcUtils.close(call);
}
}
}

(4)在out参数中使用光标

(a)申明包结构
(b)包头
(c)包体

事例:查询某个部门中所有员工的所有信息

包头:
create or replace package mypackage as
--type自定义类型empcursor
type empcursor is ref cursor;
procedure queryEmpList(dno in number,empList out empcursor);
end mypackage;
*****包体需要实现包头中生命的所有方法*******

操作:
程序包----->右键--->新建程序包--->包名设置mypackage--->确定--->编译--->ctril+s执行

包体:
create or replace package body mypackage as

procedure queryEmpList(dno in number,empList out empcursor) as
begin

open empList for select * from emp where deptno =dno;
end queryEmpList;
end mypackage;

操作:
选中mypackage-->右键-->创建主体--->编译--->ctril+s执行

查看包:
desc mypackage

注意:需要带上包名访问

(5)在应用中访问包中的存储过程
public class TestMypackage {
@Test
public void testMypackage() {
String sql = "{call MYPACKAGE.queryEmpList(?,?)}";
Connection conn = null;
CallableStatement call = null;
ResultSet rs = null;

try {
// 得到一个连接
conn = JdbcUtils.getConnection();
// 通过连接创建出statement
call = conn.prepareCall(sql);

// 对于参数赋值
call.setInt(1, 10);

// 对于输出参数,申明
call.registerOutParameter(2, OracleTypes.CURSOR);

// 执行调用
call.execute();

// 取出该部门的所有员工信息

rs = ((OracleCallableStatement) call).getCursor(2);
while (rs.next()) {
// 该员工的员工号/薪水/职位
int empno =rs.getInt("empno");
String name =rs.getString("ename");
double salary = rs.getDouble("sal");
String job = rs.getString("empjob");
System.out.println(empno +"\t"+name+"\t"+salary+"\t"+job);

}

} catch (Exception e) {
e.printStackTrace();
} finally {
JdbcUtils.close(conn);
JdbcUtils.close(call);
JdbcUtils.close(rs);
}
}
}

时间: 2024-10-23 02:19:00

Oracle学习笔记(十二)的相关文章

Oracle学习笔记十二 子程序(存储过程、自定函数)和程序包

子程序 子程序:命名的 PL/SQL 块,编译并存储在数据库中. 子程序的各个部分: 1.声明部分 2.可执行部分 3.异常处理部分(可选) 子程序的分类: 1.过程 - 执行某些操作 2.函数 - 执行操作并返回值 子程序的优点: 模块化 将程序分解为逻辑模块 可重用性 可以被任意数目的程序调用 可维护性 简化维护操作 安全性 通过设置权限,使数据更安全 存储过程 过程是用于完成特定任务的子程序. 例如: 一个购票过程可以分为很多个子过程,分别完成. 创建存储过程 创建过程的语法: CREAT

Swift学习笔记十二:下标脚本(subscript)

下标脚本就是对一个东西通过索引,快速取值的一种语法,例如数组的a[0].这就是一个下标脚本.通过索引0来快速取值.在Swift中,我们可以对类(Class).结构体(structure)和枚举(enumeration)中自己定义下标脚本的语法 一.常规定义 class Student{ var scores:Int[] = Array(count:5,repeatedValue:0) subscript(index:Int) -> Int{ get{ return scores[index];

oracle学习笔记(二)

设置归档模式(mount状态) ALTER database ARCHIVELOG; //关闭数据库 shutdown immediate //启动数据库到mount状态 startup mount alter database archivelog; //查看归档状态 archive log list; SQL> archive log list; 数据库日志模式 存档模式 自动存档 启用 存档终点 USE_DB_RECOVERY_FILE_DEST 最早的联机日志序列 1 下一个存档日志序列

虚拟机VMWare学习笔记十二 - 将物理机抓取成虚拟机

1. 安装VMware vCenter Converter Standalone Client 运行虚拟机,File -- Virtualize a Physical Machine 这时如果电脑中没有VMware vCenter Converter Standalone Client ,则会进行安装. 安装过程 之后图标会出现在桌面上,双击运行 选择连接到本地服务器,登陆 点击转换计算机 这个,可以将本地计算机抓取成虚拟机,也可以将其他可以访问的计算机(需知道管理员用户名及密码)抓取成虚拟机.

【我的Oracle学习笔记(二)】----- select语句补充

一.多表查询 多表查询是指从多个有关联的表中查询数据,其语法与单表查询类似.一般来说,多表查询的表要用连接联系起来,如果没连接,则查询结果是这多个查询表的笛卡尔积(注释1). 模拟查询雇员姓名和所在部门名称: select [雇员姓名],[部门名称] from [雇员表] a,scott,[部门表] b where a.[部门编号]=b.[部门编号]; 上例中,为每一个查询表指定了别名,便于SQL语句的书写. 模拟查询在”sales“部门工作的雇员其雇员姓名 select [雇员姓名] from

《Hibernate学习笔记十二》学生、课程、分数关系的设计与实现

<Hibernate学习笔记十二>学生.课程.分数关系的设计与实现 这个马士兵老师的Hibernate视频学习的一个题目,这里面要用到多对多.多对一的关联关系以及联合主键,因此觉得挺好的,自己写篇博文来记录下. 先考虑数据库表 1.学生表:为简单起见,只考虑了学生id和学生姓名,其中id为主键 2.课程表:为简单起见,只考虑了课程id和课程名称,其中id为主键 3.分数表 分数表有两种解决方案 3.1 第一种为:使用联合主键:student_id 和 course_id 3.2 第二种:不使用

laravel3学习笔记(十二)

原作者博客:ieqi.net ==================================================================================================== 请求反射 HTTP 协议本身是无状态性的,但是在应用中处理各种业务逻辑时我们必须要有状态的把控,这样,折中的办法就是将状态进行标记然后嵌入到 HTTP 协议的请求中,然后应用根据这些标记来进行状态的串联以及处理.所以我们就要对请求进行反射处理以获取请求信息, Lara

java jvm学习笔记十二(访问控制器的栈校验机制)

欢迎装载请说明出处:http://blog.csdn.net/yfqnihao 本节源码:http://download.csdn.net/detail/yfqnihao/4863854 这一节,我们会简单的描述一下jvm访问控制器的栈校验机制. 这节课,我们还是以实践为主,什么是栈校验机制,讲一百遍不如你自己实际的代码一下然后验证一下,下面我们下把环境搭起来. 第一步,配置系统环境.(copy吧,少年) path=%JAVA_HOME%/bin JAVA_HOME=C:/Java/jdk1.6

Oracle基础笔记十二

第十二章 用户控制权限 1.权限: 数据库安全性:系统安全性和数据安全性 系统权限: 对于数据库的权限 对象权限: 操作数据库对象的权限 2.系统权限 超过一百多种有效的权限 数据库管理员具有高级权限以完成管理任务,例如: 创建新用户 删除用户 删除表 备份表 2.1 DBA 使用 CREATE USER 语句创建用户 CREATE USER user IDENTIFIED BY   password; CREATE USER  scott IDENTIFIED BY   tiger; 2.2