`
nikoloss
  • 浏览: 33515 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

java调用存储过程的另类封装的开发日志(一)

阅读更多
        选择了存储过程,就意味着牺牲了可移植性和灵活性,换来的确是安全性和稳定性和高性能以及特定数据库的特性。不过事情往往都是相对的,正是由于plsql的灵活多变实现一个目标往往可以有很多种途径而导致说想要写出安全稳定高效并重的存储过程也不是一件容易的事。我个人觉得存储过程最方便的地方在于传统方式可能需要几次交互的信息,丢给存储过程一次就可以搞定。维护存储过程也比可能破坏设计模式和一致风格的java代码修改来说要容易的多。
        调用存储过程的花样有很多种,比如传入参数可以是基本类型,对象类型,数组类型。传出参数亦然,而且甚至可以是空参数。于是封装这个过程就不是一件容易的事情,好在经过两个星期的改进我已经成功地封装了一套模版,在工作当中稳定地运行着,现在将整个设计思想和开发的过程记录下来,包括如何一步一步的思考然后还一步一步走的弯路。可能需要很多篇博客才能描写出来,希望起到一个抛砖引玉的作用,如果大家有什么好的思想和建议还望及时与我交流相互学习促进,只要有人支持我就会写下去。
        在此之前找了一下网络上的资源,发现并没有符合我心中想法的框架,参考过springjdbc调用存储过程给了我一定的启发,但它可能考虑的东西太多导致让我觉得封装的程度还是不够理想特别对于复杂参数(对象,数组,对象数组)的传递以及多结果集的返回,我理想的是增删改查不管什么情况都是一句话调用。对于sql,hibernate已经可以做到了,但是存储过程似乎还没有,无奈只能自己动手了。
        先有一张USERS表字段三个:ID(NUMBER),NAME(VARCHAR2),BIRTHDAY(TIMESMTP)。
        JAVA的实体BEAN
package test.bean;

import java.util.Date;

public class User {
	int id;
	String name;
	Date birthday;
	public int getId() {
		return id;
	}
	public void setId(int id) {
		this.id = id;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public Date getBirthday() {
		return birthday;
	}
	public void setBirthday(Date birthday) {
		this.birthday = birthday;
	}
	
}

这种设计是为了到时候测试基本类型而使用,我个人倾向于设计bean字段全用String,因为字符型是元字型,任何别的类型都可以由它转换而来,全字符型字段也方便我们写存储过程,为什么,以后说。扯远了。。

接着就是DAO和IMP了,接口就不写出来了,首先我们要封装的是最简单的一种情况,存储过程就一个传入参数基本类型,一个返回参数游标!
package test.bean.dao;

import java.util.List;
import test.bean.User;
import test.common.ProcRunner;

public class UserDaoImp implements UserDao{
	@Override
	public List<User> getUsers(String id) {
		String sql="{CALL PKG_TEST.GETUSERS_FROM_CURSOR(?,?)}";
//这个ProcRunner类究竟是什么,现在先不关心,我们期望的效果就是这样
		return (List<User>) new ProcRunner().execute(sql,User.class);
	}
}	

这是我们期待的封装效果,那么问题就来了,如果我们仅仅在sql中使用一个“?”我们怎么才能知道它是传入还是传出?于是这个sql还需要做一点点改进。
package test.bean.dao;

import java.util.List;
import test.bean.User;
import test.common.ProcRunner;

public class UserDaoImp implements UserDao{
	@Override
	public List<User> getUsers(int id) {
		String sql="{CALL PKG_TEST.GETUSERS_FROM_CURSOR({1},{o})}";
		return (List<User>) new ProcRunner().execute(sql,User.class);
	}
}

之所以设计成{1},{2}...这种形式是因为假如这个getUsers方法参数是这样的
getUsers(int id,String name,String ...)我们就可以将多个参数设置进去,而一个
{o}代表着传出。什么?为什么不直接用 1,2,3,o?这是怕引起视觉混淆,当然这个问题其实不重要。
现在所有的重头戏就集中到了ProcRunner身上。根据我们的期望把他的架子搭出来
package test.common;

import java.sql.CallableStatement;
import java.sql.Connection;
import java.util.List;
/*
 * 这个类的设计思路我觉得应该是这样子的。首先将我们自定义的sql解析出来
 * 里面有我们指定的传入传入参数的设置和序列,它会根据我们这些符号做4件事
 * 1:设置传入参数(如果需要)
 * 2:注册传出参数(如果需要)
 * 3:执行
 * 4:获取传出结果集放到集合中返回(List<bean>|null)
 * */
public class ProcRunner {
	Connection conn;
	CallableStatement cs;
	//在面对不需要返回值的情况下,我们就不用传递一个class让它装配了
	//于是在这里设置了可变参数。
	public List<?> execute(String sql,Class<?> ...clazz){
		return null;
	}
}

接下来还要考虑就是怎么样能够把
getUsers(int id)的参数注入到这个ProcRunner里面去?否则即便我知道了要讲第一个参数设置到这个地方,可这第一个参数是什么?这样我们就要动用动态代理来帮我们把方法的参数注入到ProcRunner里面去。动态代理能够很轻易的拿到getUsers方法的所有参数,它只需要找到ProcRunner的对象就可以注入进去,这样就意味着这个ProcRunner对象不能在getUsers里去申明了,但是在DaoImp里面申明,麻烦。直接做一个父类就好了将所有要用到的对象全部申明出来就好了。这样只需要让DaoImp去继承它,就可以直接使用了,至于初始化的任务就可以全权交给动态代理了
package test.bean.dao;
import test.common.ProcRunner;
public class ProxyObjects {
	public ProcRunner pr;
}

修改daoImp层代码
package test.bean.dao;

import java.util.List;
import test.bean.User;

public class UserDaoImp extends ProxyObjects implements UserDao{
	@Override
	public List<User> getUsers(int id) {
		String sql="{CALL PKG_TEST.GETUSERS_FROM_CURSOR({1},{o})}";
		return (List<User>)pr.execute(sql,User.class);
	}
}

修改ProcRunner,使之具有参数属性。
package test.common;

import java.sql.CallableStatement;
import java.sql.Connection;
import java.util.List;

public class ProcRunner {
	Connection conn;
	CallableStatement cs;
	Object[] paras;
	String[] order;

	public Object[] getParas() {
		return paras;
	}

	public void setParas(Object[] paras) {
		this.paras = paras;
	}

	// 动态代理通过构造函数为paras赋值
	public ProcRunner(Object[] paras) {
		this.paras = paras;
	}

	public List<?> execute(String sql, Class<?>... clazz) {
		return null;
	}
}

然后创建拦截器,在此我选用的是cglib作为动态代理,因为jdk的proxy只能对实现接口类代理,而cglib是以继承的方式实现无需接口,更灵活自由。但在使用上面大同小异。
拦截器如下
package test.common.proxy;

import java.lang.reflect.Field;
import java.lang.reflect.Method;

import test.common.ProcRunner;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

public class Interceptor implements MethodInterceptor {
	private Object target;

	public Object getInstance(Object target) {
		this.target = target;
		Enhancer enhancer = new Enhancer();
		enhancer.setSuperclass(this.target.getClass());
		enhancer.setCallback(this);
		return enhancer.create();
	}

	@Override
	public Object intercept(Object obj, Method method, Object[] args,
			MethodProxy proxy) throws Throwable {
		Class<?> clazz=target.getClass().getSuperclass();
		Field field_pr=clazz.getDeclaredField("pr");
		ProcRunner pr=new ProcRunner(args);//创建对象并赋参数
		field_pr.set(target, pr);//将建立的ProcRunner对象注入到target中
		Object ret;
		ret=method.invoke(target, args);
		return ret;
	}
}

接下来从拦截器中创建daoimp就会被拦截器作用了,我确信参数已经被拦截器注入到了daoimp的procrunner对象pr中了。
现在所有的焦点就都集中到了ProcRunner身上了,他具有了我们要为存储过程设置的参数,还通过解析我们自定义的sql知道了这些参数设置的序列,实现它的思路就清晰了,还有一步准备工作就是怎么样把游标变成对象集合?完成这么一个工具类!
package test.common.utils;

import java.lang.reflect.Field;
import java.sql.ResultSet;
import java.util.ArrayList;
import java.util.List;

public class BeanUtils {
	public static List<Object> getBeans(ResultSet rs, Class<?> clazz)
			throws Exception {
		List<Object> ret = new ArrayList<Object>();
		int attrsCount = rs.getMetaData().getColumnCount();// 游标有多少列
		while (rs.next()) {
			Object bean = clazz.newInstance();
			for (int i = 1; i <= attrsCount; i++) {
				// 每列给对应的属性赋值
				String attr = rs.getMetaData().getColumnLabel(i);
				Field field = clazz.getDeclaredField(attr);
				field.setAccessible(true);
				field.set(bean, rs.getObject(i));
			}
			ret.add(bean);
		}
		return ret;
	}
}

接下来就可以完整的写出ProcRunner类了,一气呵成完成它吧!
package test.common;

import java.sql.CallableStatement;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import oracle.jdbc.OracleTypes;
import db.until.DbHelperJdbc;
import static test.common.utils.BeanUtils.*;

public class ProcRunner {
	enum ParamType{
		CURSOR,ARRAY;
	}
	Connection conn;
	CallableStatement cs;
	ResultSet rs;
	Object[] paras;//DaoImp方法上面的参数
	String[] orders;

	public Object[] getParas() {
		return paras;
	}

	public void setParas(Object[] paras) {
		this.paras = paras;
	}

	// 动态代理通过构造函数为paras赋值
	public ProcRunner(Object[] paras) {
		this.paras = paras;
	}

	/*
	 * 这个类的设计思路我觉得应该是这样子的。首先将我们自定义的sql解析出来 里面有我们指定的传入传入参数的设置和序列,它会根据我们这些符号做三件事
	 * 1:设置传入参数(如果需要) 2:注册传出参数(如果需要) 3:执行 4:获取传出结果集放到集合中返回(List<bean>|null)
	 */
	public List<?> execute(String sql, Class<?>... clazz) throws SQLException {
		List<?> ret = null;// 最终返回
		// 注册参数需要在执行之后获取,所以需要一个集合来存放注册的序号和类型
		Map<Integer, ParamType> map = new LinkedHashMap<Integer, ParamType>();
		try {
			// 获取连接
			conn = DbHelperJdbc.getInstance().getConn();
			// 获取自定义参数序列 "1,2,3,o"之类的
			orders = getOrdersFromSql(sql);
			// 准备语句,需要讲我们设置的东西重新替换成'?'
			cs = conn.prepareCall(sql.replaceAll("\\{\\w\\}", "?"));
			// 1.设置传入传出参数!
			for (int index = 0; index < orders.length; index++) {
				try {
					int order = Integer.parseInt(orders[index]);
					cs.setObject(index + 1, paras[order - 1]);
				} catch (NumberFormatException e) {
					// 转型失败说明应该是有o了说明应该是输出类型
					// 2.注册传出参数!
					cs.registerOutParameter(index + 1, OracleTypes.CURSOR);
					map.put(index + 1, ParamType.CURSOR);
				}
			}
			// 3.执行!
			cs.execute();
			// 4.获取结果集并通过传递进来的bean的class加载并放到list中返回(如果需要)!
			if (map.size() > 0) {
				ret=new ArrayList();
				int clazz_index=0;//对于传进来的bean.class是数组,我们需要用来初始化收集
				Iterator<Integer> it = map.keySet().iterator();
				while (it.hasNext()) {
					int reg_order=it.next();//注册时候的序号
					ParamType reg_type=map.get(reg_order);//注册时候的类型
					switch (reg_type) {
					case CURSOR:
						//游标怎么处理
						//先暂时写到这里吧,看看整个执行过程对不对
						rs=(ResultSet) cs.getObject(reg_order);
						ret=getBeans(rs,clazz[clazz_index]);
						break;
					}
					++clazz_index;
				}
			}
		} catch (SQLException e) {
			e.printStackTrace();
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			if (rs!=null)
				rs.close();
			if (cs != null)
				cs.close();
			if (conn != null)
				conn.close();
		}
		return ret;
	}

	// 首先我们观察传入的SQL,简化成"{X X.X({1},{2},{o})}"
	// 我们实质上是想得到其中的"1,2,o"
	public String[] getOrdersFromSql(String sql) {
		// 直接截取中括号里面的内容
		sql = sql.substring(sql.indexOf("(") + 1,
				sql.indexOf(")"));
		// 现在sql 为"{1},{2},{o}"再替换"{","}"为""
		sql = sql.replaceAll("\\{|\\}", "");
		// 现在sql为"1,2,o" 按","分割返回
		return sql.split(",");
	}
}

现在激动人心的时刻到了,测试是惊险的。
先往表里添加记录
   INSERT INTO USERS VALUES('1','Billy',sysdate);
   INSERT INTO USERS VALUES('2','Lily',sysdate);
   INSERT INTO USERS VALUES('3','Rowland',sysdate);

存储过程:
CREATE OR REPLACE PACKAGE BODY PKG_TEST IS
   PROCEDURE GETUSERS_FROM_CURSOR(I_ID IN NUMBER,O_RES OUT SYS_REFCURSOR) AS
     BEGIN
       OPEN O_RES FOR SELECT 
         ID AS "id",
         NAME AS "name",
         BIRTHDAY AS "birthday"
         FROM USERS WHERE ID=I_ID;
     END;
END PKG_TEST;

package test;

import java.util.List;

import test.bean.User;
import test.bean.dao.UserDao;
import test.bean.dao.UserDaoImp;
import test.common.proxy.Interceptor;

public class Test {
	public static void main(String[] args) throws Exception {
		UserDao ud=(UserDao) new Interceptor().getInstance(new UserDaoImp());
		List<User> list=ud.getUsers(1);
		System.out.println("beans count:"+list.size());
		for(User user:list){
			System.out.println("id="+user.getId());
			System.out.println("name="+user.getName());
			System.out.println("birthday"+user.getBirthday());
		}
	}
}

结果呢?报错了
java.lang.IllegalArgumentException: Can not set java.lang.Integer field test.bean.User.id to java.math.BigDecimal
这么看来应该是类型错误,如果我们仅仅只是rs.getObject就很容易出现这种问题,应该要使用rs.getInt去赋值bean的id,现在知道我说我倾向于在Bean时候使用字符类型了吧,有人会说那我java端要拿去做计算不方便,我说,既然选择了使用存储过程就应该把业务逻辑全权交给存储过程java端就只需要取出来展示一下,或者传进数据库存入一下。不应该出现java端还有复杂计算。针对一些展示格式上面特别的小的拼接和运算,那么交给前台js去处理就好了,js本身也是弱类型的。所以解决的思路有两个,那么在BeanUtil里拿到bean属性的时候去判断一下类型,这么一来你至少需要8个基本类型+字符型+Date+...,我是比较懒了,所以我直接把bean的属性都改成string算了。
那么相应的存储过程就需要全部返回字符型了
CREATE OR REPLACE PACKAGE BODY PKG_TEST IS
   PROCEDURE GETUSERS_FROM_CURSOR(I_ID IN NUMBER,O_RES OUT SYS_REFCURSOR) AS
     BEGIN
       OPEN O_RES FOR SELECT 
         to_char(ID) AS "id",
         NAME AS "name",
         to_char(BIRTHDAY,'yyyy-mm-dd') AS "birthday"
         FROM USERS WHERE ID=I_ID,ID;
     END;
END PKG_TEST;

再来测试一下,结果如下:
beans count:1
id=1
name=Billy
birthday=2012-11-24
测试完了一个参数的情况,我们测试一下两个参数看看吧
package test.bean.dao;

import java.util.List;
import test.bean.User;

public class UserDaoImp extends ProxyObjects implements UserDao {
	public List<User> getUsers(int id,String name) throws Exception {
		return (List<User>) pr.execute("{CALL PKG_TEST.GETUSERS_FROM_CURSOR({1},{2},{o})}", User.class);
	}
}

   PROCEDURE GETUSERS_FROM_CURSOR(I_ID IN NUMBER,I_NAME IN VARCHAR2,O_RES OUT SYS_REFCURSOR) AS
     BEGIN
       OPEN O_RES FOR SELECT 
         to_char(ID) AS "id",
         NAME AS "name",
         to_char(BIRTHDAY,'yyyy-mm-dd') AS "birthday"
         FROM USERS WHERE ID=ID AND NAME=I_NAME;
     END; 

测试代码只需要改动一个地方
List<User> list=ud.getUsers(1,"Bily");
看看结果?
beans count:0
什么?没报错?又没查出来?什么情况?赶紧查查数据库,哎呀我去!数据库里是"Billy"
实在是太粗心了。改成Billy之后正常。
我们再来测试一下无参的情况
package test.bean.dao;

import java.util.List;
import test.bean.User;

public class UserDaoImp extends ProxyObjects implements UserDao {
	public List<User> getUsers()throws Exception {
		return (List<User>) pr.execute("{CALL PKG_TEST.GETUSERS_FROM_CURSOR({o})}", User.class);
	}
}

存储过程只需要将传入参数删掉,where语句删掉就不贴出来了。结果如下
beans count:3
id=1
name=Billy
birthday=2012-11-24
id=2
name=Lily
birthday=2012-11-24
id=3
name=Rowland
birthday=2012-11-24
OK,大功告成。看来我们达到了目标了,接下来的任务就是不断地对ProcRunner进行开发,使之兼容各种类型的传入传出。这一篇解决了游标,下一篇咱们来解决数组。

总结:其实整个过程并没有使用太深奥超前的技术,尽量不借助第三方包而完成整个过程。最麻烦的地方在于多样的类型,如果想要真正封装成一套框架的话,那么对于细节的处理就不能像我这么大条。好在终于是完成了它。今后使用存储过程就显得很方便了。
1
0
分享到:
评论

相关推荐

    java调用存储过程(含out参数)

    总之,调用存储过程是Java开发中处理数据库操作的重要环节。理解如何设置`IN`和`OUT`参数,并正确地执行和获取结果,将有助于提高代码的效率和可维护性。希望这个详细讲解对你有所帮助,如果你在实践中遇到任何问题...

    JAVA调用ORACLE存储过程通用类

    在Java开发中,经常会遇到需要与数据库交互的情况,特别是当涉及到复杂的业务逻辑时,利用数据库的存储过程可以有效地提高程序性能并简化代码结构。本篇文章将详细介绍一个用于Java应用程序中调用Oracle存储过程的...

    java调用存储过程同时返回值和多个table

    java调用存储过程,支持获取return值,output返回值,以及查询的表数据,表数据允许有多个查询结果集

    详解java调用存储过程并封装成map

    Java调用存储过程并封装成Map是Java开发中一个常见的需求,通过本文,我们将详细介绍如何使用Java调用存储过程并将结果封装成Map对象,以便于后续的数据处理。 一、为什么需要调用存储过程 在实际开发中,我们经常...

    java调用oracle存储过程返回结果集,Record,cursor参照.pdf

    本文介绍了Java调用Oracle存储过程返回结果集Record和Cursor的相关知识点,包括创建Type、PACKAGE和PROCEDURE,Java代码中调用存储过程,并获取结果集。这些知识点对深入了解Oracle存储过程和Java调用Oracle存储过程...

    java调用oracle存储过程或者函数

    调用Oracle函数的过程与调用存储过程类似,但创建CallableStatement时的SQL语句略有不同。由于函数会返回一个值,所以格式通常是`{? = call function_name(?, ?, ...)}`。在执行后,通过`CallableStatement....

    C++库封装JNI接口-实现java调用c++

    在跨平台的软件开发中,有时我们需要在Java和C++之间进行交互,这通常是由于性能需求、使用已有的C++库或特定硬件接口的原因。Java Native Interface (JNI) 是Java平台提供的一种机制,允许Java代码和其他语言写的...

    java调用C#封装的dll方法

    为了实现Java与C#之间的交互,一种常用的方式是通过调用C#封装的DLL文件来完成特定功能。这种方法涉及到.NET Framework与Java平台之间的互操作性问题。其中一种方案是使用Java Native Access (JNA) 来调用通过C++...

    java调用存储过程小结.pdf

    Java 调用存储过程小结 Java 调用存储过程是指在 Java 程序中调用数据库中的存储过程,以便实现复杂的业务逻辑和数据处理。存储过程是指保存在数据库并在数据库端执行的程序,可以使用特殊的语法在 Java 类中调用...

    Java实现调用MySQL存储过程详解

    3. **调用存储过程**:使用`CallableStatement`对象来执行存储过程。对于带参数的存储过程,需要设置参数并执行`executeProcedure()`。 ```java String procedureName = "{call findAllBook(?) }"; // 例如,调用...

    java调用存储过程源码 sql数据库

    总的来说,Java调用存储过程是数据库交互的一个重要环节,理解其原理和操作流程对于开发高效、可靠的数据库应用至关重要。在实际项目中,根据具体需求选择合适的设计策略,既能保证功能实现,又能兼顾性能和维护性。

    Java调用oracle存储过程通过游标返回临时表

    在Oracle中,调用存储过程通常以`BEGIN`和`END`包围,使用`CALL`关键字。例如: ```java String callProc = "{call package_name.procedure_name(?, ?)}"; PreparedStatement pstmt = connection.prepareCall...

    JAVA调用存储过程

    ### JAVA调用存储过程知识点详解 #### 一、无结果集返回的存储过程调用 在Java中调用不返回结果集的存储过程时,主要使用`CallableStatement`对象配合预定义的SQL语法来实现。这里以SQL Server 2000数据库为例,并...

    java调用shell脚本完美封装

    本文将详细介绍如何在Java中完美封装shell脚本的调用方法,并提供相关的测试示例。 1. **Java与Shell脚本的桥梁:Runtime类和ProcessBuilder类** Java提供了两种主要的方式来调用外部程序,包括Shell脚本:`...

    java 调用存储过程

    在数据库管理中,存储过程是一种预编译的SQL语句集合,它可以封装一系列的操作,比如数据查询、更新等,提供了一种高效且安全的方式来...在开发过程中,结合这些资源可以更好地理解和实践Java调用存储过程的整个流程。

    java调用sqlserver存储过程.pdf

    4. Java 调用存储过程 在 Java 应用程序中,可以使用 JDBC(Java Database Connectivity)来调用 SQL Server 数据库中的存储过程。在本例中,使用了 `java.sql` 包中的 `CallableStatement` 对象来调用 `InsertUser...

    JAVA调用存储过程.txt

    ### Java调用存储过程的基本步骤 #### 一、加载数据库驱动 在Java程序中调用数据库存储过程的第一步是加载相应的数据库驱动。例如,在本例中使用的Oracle数据库驱动通过以下代码加载: ```java Class.forName(...

    springboot mybatis 动态调用oracle存储过程,通过存储过程名称,就能动态调用存储过程、java动态调用or

    能不能写个动态的业务,只输入存储过程名称,自动...只写一个通用方法,就可以调用所有的存储过程。只根据输入不同的存储过程名称、参数内容,自动调用不同的存储过程。 已经使用在多个项目中 全开源项目 请放心下载

    Java调用存储过程

    Java调用存储过程是数据库操作中的常见任务,特别是在复杂业务逻辑和数据处理中。存储过程是一种预编译的SQL语句集合,可以在数据库服务器端执行,提高了效率并减少了网络通信量。本文将详细讲解Java如何调用存储...

    调用存储过程(java)

    JDBC提供了一组接口和类,使得Java程序能够执行SQL语句,包括调用存储过程。主要涉及的接口有`java.sql.CallableStatement`,它是`PreparedStatement`的子接口,用于执行数据库的存储过程。 3. **CallableStatement...

Global site tag (gtag.js) - Google Analytics