【本文介绍】

熟悉了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)——存储过程的更多相关文章

  1. Oracle入门4-REF Cursor

    Oracle入门4-REF Cursor 转自:http://blog.sina.com.cn/s/blog_55dbebb00100gxsc.html 自:http://blog.csdn.net/ ...

  2. Dapper完美兼容Oracle,执行存储过程,并返回结果集。

    Dapper完美兼容Oracle,执行存储过程,并返回结果集. 这个问题,困扰了我整整两天. 刚刚用到Dapper的时候,感觉非常牛掰.特别是配合.net 4.0新特性dynamic,让我生成泛型集合 ...

  3. Oracle中执行存储过程call和exec区别

    Oracle中执行存储过程call和exec区别 在sqlplus中这两种方法都可以使用: exec pro_name(参数1..); call pro_name(参数1..); 区别: 1. 但是e ...

  4. Oracle job procedure 存储过程定时任务

    Oracle job procedure 存储过程定时任务 oracle job有定时执行的功能,可以在指定的时间点或每天的某个时间点自行执行任务. 一.查询系统中的job,可以查询视图 --相关视图 ...

  5. Oracle dbms_lock.sleep()存储过程使用技巧-场景-分析-实例

    <Oracle dbms_lock.sleep()存储过程使用技巧>-场景-分析-实例 摘要:今天是2014年3月10日,北京,雾霾,下午组织相关部门开会.会议的结尾一名开发工程师找到了我 ...

  6. oracle 中的存储过程

      oracle 中的存储过程 --oracle 中的存储过程, --不带任何参数的 CREATE OR REPLACE PROCEDURE PRO_TEST AS -- AS 和is 没有任何区别 ...

  7. 【转】Oracle job procedure 存储过程定时任务

    原文:Oracle job procedure 存储过程定时任务 oracle job有定时执行的功能,可以在指定的时间点或每天的某个时间点自行执行任务. 一.查询系统中的job,可以查询视图 --相 ...

  8. oracle函数和存储过程有什么区别

    oracle函数和存储过程有什么区别 1. 返回值的区别,函数有1个返回值,而存储过程是通过参数返回的,可以有多个或者没有 2.调用的区别,函数可以在查询语句中直接调用,而存储过程必须单独调用. 函数 ...

  9. Oracle通用分页存储过程的创建与使用

    Oracle通用分页存储过程的创建与使用 1.创建Oracle包的定义.使用 REF CURSOR 数据类型来处理 Oracle 结果集.REF CURSOR 是一个指向 PL/SQL 查询所返回的结 ...

  10. 用sql语句导出oracle中的存储过程和函数

    用sql语句导出oracle中的存储过程和函数: SET echo off ; SET heading off ; SET feedback off ; SPOOL 'C:/PRC.SQL' repl ...

随机推荐

  1. fontDialog-字体对话框和colorDialog-颜色对话框

    private void button1_Click(object sender, EventArgs e) { DialogResult dr = fontDialog1.ShowDialog(); ...

  2. Java范型之T extends Comparable<? super T>

    在观察Java源码的时候,发现了这么一个写法T extends Comparable<? super T>.不禁纳闷为什么要这么写呢?有什么好处吗,extends和super在这里的作用着 ...

  3. Btrace的使用方法

    本文基于<深入理解Java虚拟机:JVM高级特性与最佳实践 第2版> 写在前面: Btrace有很多用法,比如说性能监视,连接泄露,内存泄漏,多线程竞争,而本文说的只是最基本的应用打印调用 ...

  4. openresty package.path require 报错

    在文件中 package.path = '/usr/local/share/lua/5.1/?.lua;/usr/local/openresty/lualib/resty/?.lua;' packag ...

  5. 实现Easyui 可编辑表格

    一.前端框架使用的easyui框架 二.后端使用的是jfinal 三.效果图 四.html代码 <div id="table_tree" class="easyui ...

  6. C++ 类的构造函数使用规则

    //类的构造函数使用规则 #define _CRT_SECURE_NO_WARNINGS #include<iostream> using namespace std; class Poi ...

  7. spark 系列文章汇总

    源码导读 spark 源码导读1 从spark启动脚本开始 spark 源码导读2 进一步窥探Master.Worker启动及通信机制 spark 源码导读3 进一步理解脚本调用关系 spark 源码 ...

  8. ti8168平台的tiler memory

    DM8168 DMM/TILER简介 1.概述 如图4-1,DMM定位在SDRAM控制器的前端,是所有initiator产生的内存存取的接口. 动态内存管理器DMM,是一个专门的管理模块,广义上说,包 ...

  9. 关于Java中的HashMap的深浅拷贝的测试与几点思考

    0.前言 工作忙起来后,许久不看算法,竟然DFA敏感词算法都要看好一阵才能理解...真是和三阶魔方还原手法一样,田园将芜,非常可惜啊. 在DFA算法中,第一步是需要理解它的数据结构,在此基础上,涉及到 ...

  10. 禁止Apache显示目录索引的常见方法

    禁止Apache显示目录索引,禁止Apache显示目录结构列表,禁止Apache浏览目录,这是网上提问比较多的,其实都是一个意思.下面说下禁止禁止Apache显示目录索引的常见的3种方法. 要实现禁止 ...