oracle入门(7)——存储过程
【本文介绍】
熟悉了PL/SQL语法后,实现java调用oracle存储过程才是主要目的。本文将介绍如何写存储过程,java如何调用存储过程。
【存储过程介绍】
抛开专业的描述,存储过程就是在数据库里面写了一些函数,我们在代码(如java)里面调用这些函数实现对数据库的操作,避免了数据库对SQL语句的解析,对于需要发送多条SQL语句才能完成的数据库操作功能来说,速度上升了一个档次。不过程序在操纵数据库这一块 的维护性会降低,因为存储过程是写在数据库,不是写在程序里。
【如何写有能传值 并且 有返回值(非返回列表)存储过程】
非返回列表的存储过程比较容易,直接在函数名后面带参数就好,同时赋予参数的类型。
下面举一个例子:
存储过程代码:
CREATE OR REPLACE
procedure serachUserMethod(para1 IN VARCHAR2, para2 OUT VARCHAR2)
as
BEGIN
select "user"."name" into para2 from "user" where "user"."name"=para1;
end;
java代码:
package com.zjm.www.test; import java.sql.CallableStatement;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.sql.Types;
import java.util.Date; import org.junit.After;
import org.junit.Before;
import org.junit.Test; public class test { private static String driverclass="oracle.jdbc.driver.OracleDriver";
// 本地
private static String url="jdbc:oracle:thin:@localhost:1521:orcl";
private static String username="test";
private static String password="Aaa38324836"; private static String sql="";
private static Connection conn = null;
private static Statement stmt = null;
private static ResultSet rs = null;
private static CallableStatement proc = null;; @Before
public void before(){
try {
Class.forName(driverclass).newInstance(); //加载驱动
conn=DriverManager.getConnection(url,username,password); //获得连接
stmt=conn.createStatement();
} catch (Exception e) {
e.printStackTrace();
} } @After
public void after(){
try {
if(conn != null){
conn.close();
}
} catch (Exception e) {
e.printStackTrace();
}
try {
if(stmt != null){
stmt.close();
}
} catch (Exception e) {
e.printStackTrace();
}
try {
if(rs != null){
rs.close();
}
} catch (Exception e) {
e.printStackTrace();
} } /**
* 测试调用存储过程用时
* @throws SQLException
*/
@Test
public void getDateByMethod(){ int pre = (int) System.currentTimeMillis(); try {
// 调用存储过程,问号个数代表参数存储过程的个数,且顺序要一一对应
proc = conn.prepareCall("{ call serachUserMethod(?,?) }");
// 传值
proc.setString(1, "AAAFB7E4B4D14475AD994310EF62EBA7");
// 注册返回值
proc.registerOutParameter(2, Types.VARCHAR);
// 提交
proc.execute();
// 取出返回值
System.out.println(proc.getString(2)); } catch (SQLException e) {
e.printStackTrace();
} finally {
proc.close();
}
int post=(int) System.currentTimeMillis();
System.out.println("测试调用存储过程用时"+(post-pre));
}
}
注意,这里的proc.getString(2)中的数值2并非任意的,而是和存储过程中的 参数 列对应的,如果out是在第一个位置,那就是proc.getString(1),如果是第三个位置,就是proc.getString(3),当然也可以同时有多个返回值,那就是再多加几个out参数了。
【返回列表的存储过程】
通过上面的例子,返回普通的int ,Stirng 类型的参数还是没问题的,可是这永远不能满足需求,我们一般要的是返回数据库里一个表的所有属性?当然可以。
第一步:建包(游标)。
存储过程是不能直接返回一个对象(这个对象有N个属性)的,但它有另外一种方式,通过建立一个”指针“一样的”游标“ 指向某些数据,最后返回这个 ”游标“ ,我们就能顺着这个游标拿到我们想要的数据了。
create or replace package testpackage as
type Test_CURSOR is ref cursor;
end testpackage;
第二步:创建存储过程。
注意:返回参数的类型为我们刚刚建立的 ”游标“ 类型
create or replace procedure testc(p_cursor out testpackage.Test_CURSOR)
is
begin
open p_cursor for select * from T_AP_ZA_LYT_GNLK;
end testc;
第三步:java代码:
package com.zjm.www.test; import java.sql.CallableStatement;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.sql.Types;
import java.util.Date; import org.junit.After;
import org.junit.Before;
import org.junit.Test; public class test { private static String driverclass="oracle.jdbc.driver.OracleDriver";
// 本地
private static String url="jdbc:oracle:thin:@localhost:1521:orcl";
private static String username="test";
private static String password="Aaa38324836"; private static String sql=""; // 记得表名要用""括起来
private static Connection conn = null;
private static Statement stmt = null;
private static ResultSet rs = null;
private static CallableStatement proc = null;; @Before
public void before(){
try {
Class.forName(driverclass).newInstance(); //加载驱动
conn=DriverManager.getConnection(url,username,password); //获得连接
stmt=conn.createStatement();
} catch (Exception e) {
e.printStackTrace();
} } @After
public void after(){
try {
if(conn != null){
conn.close();
}
} catch (Exception e) {
e.printStackTrace();
}
try {
if(stmt != null){
stmt.close();
}
} catch (Exception e) {
e.printStackTrace();
}
try {
if(rs != null){
rs.close();
}
} catch (Exception e) {
e.printStackTrace();
} }
@Test
public void getDateByMethod2(){
int pre = (int) System.currentTimeMillis(); try {
proc = conn.prepareCall("{ call testc(?) }");
// 注册返回值参数,注意是游标类型
proc.registerOutParameter(1,oracle.jdbc.OracleTypes.CURSOR);
proc.execute();
// 返回的数据存储在一个ResultSet里面
ResultSet rs = (ResultSet)proc.getObject(1);
while(rs.next())
{
// getString(...)里面填的对应数据库 表 的字段名
System.out.println("<tr><td>" + rs.getString("id") + "</td><td>"+rs.getString("name")+"</td></tr>");
}
proc.close();
} catch (SQLException e) {
e.printStackTrace();
} int post=(int) System.currentTimeMillis(); System.out.println("测试调用存储过程用时"+(post-pre));
} }
oracle入门(7)——存储过程的更多相关文章
- Oracle入门4-REF Cursor
Oracle入门4-REF Cursor 转自:http://blog.sina.com.cn/s/blog_55dbebb00100gxsc.html 自:http://blog.csdn.net/ ...
- Dapper完美兼容Oracle,执行存储过程,并返回结果集。
Dapper完美兼容Oracle,执行存储过程,并返回结果集. 这个问题,困扰了我整整两天. 刚刚用到Dapper的时候,感觉非常牛掰.特别是配合.net 4.0新特性dynamic,让我生成泛型集合 ...
- Oracle中执行存储过程call和exec区别
Oracle中执行存储过程call和exec区别 在sqlplus中这两种方法都可以使用: exec pro_name(参数1..); call pro_name(参数1..); 区别: 1. 但是e ...
- Oracle job procedure 存储过程定时任务
Oracle job procedure 存储过程定时任务 oracle job有定时执行的功能,可以在指定的时间点或每天的某个时间点自行执行任务. 一.查询系统中的job,可以查询视图 --相关视图 ...
- Oracle dbms_lock.sleep()存储过程使用技巧-场景-分析-实例
<Oracle dbms_lock.sleep()存储过程使用技巧>-场景-分析-实例 摘要:今天是2014年3月10日,北京,雾霾,下午组织相关部门开会.会议的结尾一名开发工程师找到了我 ...
- oracle 中的存储过程
oracle 中的存储过程 --oracle 中的存储过程, --不带任何参数的 CREATE OR REPLACE PROCEDURE PRO_TEST AS -- AS 和is 没有任何区别 ...
- 【转】Oracle job procedure 存储过程定时任务
原文:Oracle job procedure 存储过程定时任务 oracle job有定时执行的功能,可以在指定的时间点或每天的某个时间点自行执行任务. 一.查询系统中的job,可以查询视图 --相 ...
- oracle函数和存储过程有什么区别
oracle函数和存储过程有什么区别 1. 返回值的区别,函数有1个返回值,而存储过程是通过参数返回的,可以有多个或者没有 2.调用的区别,函数可以在查询语句中直接调用,而存储过程必须单独调用. 函数 ...
- Oracle通用分页存储过程的创建与使用
Oracle通用分页存储过程的创建与使用 1.创建Oracle包的定义.使用 REF CURSOR 数据类型来处理 Oracle 结果集.REF CURSOR 是一个指向 PL/SQL 查询所返回的结 ...
- 用sql语句导出oracle中的存储过程和函数
用sql语句导出oracle中的存储过程和函数: SET echo off ; SET heading off ; SET feedback off ; SPOOL 'C:/PRC.SQL' repl ...
随机推荐
- 专题实验 Toad 用户的创建与管理( 包括 role 等 )
1. 用户登录数据库 是否可以通过操作系统权限来登录数据库, $ORACLE_HOME/network/admin/sqlnet.ora 这个文件中设置, 如果增加参数sqlnet.authentic ...
- 【转】VC调试的时候 “没有调试信息,未加载符号”
概述调试是一个程序员最基本的技能,其重要性甚至超过学习一门语言.不会调试的程序员就意味着他即使会一门语言,却不能编制出任何好的软件.这里我简要的根据自己的经验列出调试中比较常用的技巧,希望对大家有用. ...
- BZOJ 3922 - Karin的弹幕
Karin的弹幕 Problem's Link ---------------------------------------------------------------------------- ...
- PHP 汉字转成拼音
<?php class ZH{ /** * 将字符串转化为拼音 */ function Pinyin($_String, $_Code='gb2312') { $_DataKey =" ...
- TF Boys (TensorFlow Boys ) 养成记(三): TensorFlow 变量共享
上次说到了 TensorFlow 从文件读取数据,这次我们来谈一谈变量共享的问题. 为什么要共享变量?我举个简单的例子:例如,当我们研究生成对抗网络GAN的时候,判别器的任务是,如果接收到的是生成器生 ...
- 第二百七十二节,Tornado框架-iframe标签框架伪造ajax
Tornado框架-iframe标签框架伪造ajax html <!DOCTYPE html> <html> <head lang="en"> ...
- 如果输入参数采用“指针传递”,那么加 const 修饰可以防止意外地改动该指针,起 到保护作用
如果输入参数采用“指针传递”,那么加 const 修饰可以防止意外地改动该指针,起 到保护作用. #include <iostream> /* run this program using ...
- php的form中元素name属性相同时的取值问题
php的form中元素name属性相同时的取值问题:修改元素的名称,在名称后面加上 '[]',然后取值时即可得array()数组. 一.以复选框为例: <html> <head> ...
- c# http请求添加cookie
CookieCollection cookList = new CookieCollection(); cookList.Add(new Cookie("cf_clearance" ...
- C语言位运算+实例讲解(转)
按位或 按位与 按位异或 按位取反 左移右移 C语言位运算 有6种: &, | , ^(亦或), >(右移). 注意:参与位运算的元素必须是int型或者char型,以补码形式出现. 按位 ...