- 浏览: 199337 次
- 性别:
- 来自: 广州
文章分类
最新评论
-
feihumingyue:
nice 很好啊
JSF中如何使用FacesContext类 -
wgcniler:
请问如果传到存储过程的参数是一个嵌套表的话该怎么写?自定义的o ...
spring中调用存储过程 -
wgcniler:
请问如果传到存储过程的参数是ARRAY,但ARRAY的元素不是 ...
spring中调用存储过程 -
bengan:
谢谢楼上的提示
关于出现僵尸信号SIGBAT或者EXC_BAD_ACCESS的解决方案 -
gypgyp:
用xcode的菜单:product/profile,弹出窗口中 ...
关于出现僵尸信号SIGBAT或者EXC_BAD_ACCESS的解决方案
org.springframework.jdbc.object.StoredProcedure是对应存储过程调用的操作对象,它通过其父类org.springframework.jdbc.object.SqlCall获得相应的底层API支持(CallableStatementCreator), 然后在此基础之上构建了调用存储过程的执行方法。
StoredProcedure是抽象类,所以需要实现相应子类以封装对特定存储过程的调用,还记得我们在讲解JdbcTemplate调用存储过程时候定义的存储过程吗?
CREATE PROCEDURE CountTable(IN tableName varchar(1000),OUT sqlStr varchar(1000) , INOUT v INT) BEGIN set @flag = v; set @sql = CONCAT('select count(*) into @res from ' , tableName , ' where ACTIVE_FLAG=?'); PREPARE stmt FROM @sql; EXECUTE stmt using @flag; DEALLOCATE PREPARE stmt; set v = @res; set sqlStr = @sql; END
通过继承StoredProcedure,我们可以为该存储过程的调用提供一个对应的操作对象:
public class CountTableStoredProcedure extends StoredProcedure { private static final String PROCEDURE_NAME = "CountTable"; public static final String IN_PARAMETER_NAME = "tableName"; public static final String OUT_PARAMETER_NAME = "sqlStr"; public static final String INOUT_PARAMETER_NAME = "v"; public CountTableStoredProcedure(DataSource dataSource) { super(dataSource,PROCEDURE_NAME); // setFunction(true); declareParameter(new SqlParameter(IN_PARAMETER_NAME,Types.VARCHAR)); declareParameter(new SqlOutParameter(OUT_PARAMETER_NAME,Types.VARCHAR)); declareParameter(new SqlInOutParameter(INOUT_PARAMETER_NAME,Types.INTEGER)); compile(); } public CountTableResult doCountTable(String tableName,Integer v) { Map paraMap = new HashMap(); paraMap.put(IN_PARAMETER_NAME, tableName); paraMap.put(INOUT_PARAMETER_NAME, v); Map resultMap = execute(paraMap); CountTableResult result = new CountTableResult(); result.setSql((String)resultMap.get(OUT_PARAMETER_NAME)); result.setCount((Integer)resultMap.get(INOUT_PARAMETER_NAME)); return result; } }
关于该存储过程操作对象,部分细节我们有必要关注一下:
-
存储过程操作对象对应的SQL是存储过程的名称,而不是真正意义上的SQL语句,当我们调用compile方法的时候, StoredProcedure的父类SqlCall会根据你提供的存储过程名称拼装真正意义上的符合SQL92标准的存储过程调用语句, 类似于“{ call CountTable(?,?,?) }”的形式。
因为我们的CountTableStoredProcedure只针对CountTable存储过程调用,所以,该存储过程的名称我们在类一开始就声明为常量:
private static final String PROCEDURE_NAME = "CountTable";
如果有多个存储过程的参数顺序相同,结果处理也一样的话,你也可以将存储过程的名称声明为变量,这完全要取决于具体的应用场景。 -
在构造方法中,我们将“setFunction(true);”注释掉了,因为我们调用的CountTable不是一个Function, 如果你要调用的存储过程类型为Function的话,你需要通过该方法将“function”的值设置为true,以告知StoredProcedure在处理调用的时候要区别对待。
-
在complie之前通过declareParameter声明参数,这几乎是雷打不动的惯例,不过,在StoredProcedure中使用declareParameter的时候却要有所注意了:
-
针对存储过程参数类型为IN,OUT和INOUT不同,declareParameter接受的参数类型也应该是SqlParameter,SqlOutParameter和SqlInOutParameter;
-
SqlParameter,SqlOutParameter和SqlInOutParameter的相应实例在构造的时候,必须指定对应的参数名称,因为在调用存储过程的时候, 需要根据名称传入参数,更需要根据名称取得调用结果;
-
-
StoredProcedure提供了execute方法执行存储过程调用,通过Map的形式传入调用所需要的IN或者INOUT类型的参数值, 所以,在构建参数Map的时候,该Map中的Key应该与declareParameter时候声明的参数名称相同; 另外,execute执行后返回的结果也是Map形式,从该结果Map中取得具体的结果值的时候,也是通过declareParameter中声明的OUT/INOUT参数名作为key来获取的, 所以,这就是我们将各个参数的名称在类定义的开始声明为常量的原因:
public static final String IN_PARAMETER_NAME = "tableName"; public static final String OUT_PARAMETER_NAME = "sqlStr"; public static final String INOUT_PARAMETER_NAME = "v";
无论是输入参数Map还是输出参数结果对应的Map,他们中的Key应该与通过declareParameter方法声明的参数名称一一对应。
通过扩展StoredProcedure,我们不但封装了参数的声明和结果的提取,我们还为调用方提供了强类型的调用方法, 现在,调用方可以通过doCountTable方法的强类型参数声明传入参数值,并取得强类型的CountTableResult对象作为结果,而不是泛泛的一个Map。
对于存储过程的调用者来说,它的代码现在可以简洁到两行代码:
// DataSource dataSource = ...; CountTableStoredProcedure storedProcedure = new CountTableStoredProcedure(dataSource); CountTableResult result = storedProcedure.doCountTable("tableName",1); ...
漂亮多了,不是吗?
StoredProcedure提供了两个execute方法执行存储过程的调用,一个就是我们刚才使用的通过Map提供输入参数的execute方法,另一个则是使用ParameterMapper类型提供输入参数的execute方法。 那么,为什么要提供这个使用ParameterMapper类型提供输入参数的execute方法那?
ParameterMapper定义的callback方法暴露了相应的Connection,如果说在构造输入参数列表的时候,必须用到Connection的话, ParameterMapper恰好可以提供支持。比如,Oracle中定义的一存储过程,接收数组类型作为参数,而在oracle中,你只能通过Oracle.sql.ARRAY和相应的Oracle.sql.ArrayDescriptor来定义数组类型的参数, ARRAY和ArrayDescriptor都需要用到相应的Connection进行构造。所以,对于Oracle中需要使用数组传入参数的存储过程来说,我们可以通过如下类似代码进行调用:
public class OracleStoredProcedure { ... public Map call(...) { ParameterMapper paramMapper = new ParameterMapper(){ public Map createMap(Connection connection) throws SQLException { Map inMap = new HashMap(); ... Integer[] params = new Integer[]{new Integer(1),new Integer(2)}; ArrayDescriptor desc = new ArrayDescriptor("numbers", connection); ARRAY nums = new ARRAY(desc, connection, params); inMap.put("ArrayParameterName", nums); ... return inMap; }}; return execute(paramMapper); } }
当然啦,我们的CountTableStoredProcedure在调用存储过程的时候也可以使用ParameterMapper传入相应的调用参数,只不过,ParameterMapper的createMap方法暴露的Connection对于我们来说没有太大用处罢了。
评论
自定义的oracle对象类型为:
create or replace
TYPE passenger_trip_condition AS OBJECT(
ticket_no VARCHAR2(20),
departure_date VARCHAR2(10),
departure_airport_code VARCHAR2(3),
arrival_airport_code VARCHAR2(3)
);
我自定义了一个嵌套表:
create or replace
TYPE psg_trip_condition_table AS TABLE OF passenger_trip_condition;
存储过程为:
create or replace
PROCEDURE update_psg_trip_proc (V_ARRAY IN psg_trip_condition_table) IS
BEGIN
...
END update_psg_trip_proc;
以上的存储过程应该怎么用spring调用?
如果是个CLOB型的参数该怎么获取,比如
declareParameter(new SqlInOutParameter (INOUT_PARAMETER_NAME,Types.CLOB));
原文为:(String)resultMap.get(OUT_PARAMETER_NAME)
发表评论
-
spring整合struts的3种配置方式
2011-08-08 01:43 866实例讲解spring整合struts的几种方式 1,使用Sp ... -
抽象语法树(AST)
2010-01-16 04:02 9571原文出自:http://blog.csdn.net/zhouh ... -
java虚拟机参数
2009-11-19 01:21 1000下面的讨论以Windows平台的Sun MicroSystem ... -
JVM 堆内存(heap)设置选项
2009-11-18 22:17 3011JVM 堆内存(heap)设置选项 参数格式 说 ... -
java堆栈
2009-11-15 11:12 1532java中堆栈(stack)和堆(heap) 一、堆栈(st ... -
Tomcat启动过程
2009-04-01 11:37 1084今天在独立的Tomcat中部 ... -
JSF如何使用ExternalContext类 2
2008-11-04 14:51 23053.3.8 获取CookiegetRequestCookie ... -
JSF如何使用ExternalContext类 1
2008-11-04 14:42 2219使用ExternalContext类提供 ... -
JSF中如何使用FacesContext类
2008-11-04 14:26 2163在Faces API中有两个类是要经常使用的. 一个是Face ... -
JSF Mbean
2008-09-04 20:23 1173首先从Model1中的JavaBean说起,大家知道,Mode ... -
JSF FacesContext 详解 三
2008-08-28 15:48 40283.3.6 访问Request对象里的参数名和值 getRe ... -
JSF FacesContext 详解 二
2008-08-28 15:47 2637JSF FacesContext 详解 二 3. ... -
JSF FacesContext 详解 一
2008-08-28 15:45 2163JSF FacesContext 详解 一 ... -
一个jsf的face-config.xml看不懂
2008-08-25 16:46 3480刚学习jsf,公司一个jsf工程里的face-config.x ... -
jar命令详解
2008-08-20 00:17 1151自己学习记录的资料 1. ... -
jmx使用实例(待加注析正式发布)
2008-08-20 00:00 0JMX 在JDK 1.5中已经是J2SE的一个标准组成部分了。 ... -
最近一个工作要用到jmx所以特意找了一些资料了解jmx,好的给大家分享
2008-08-19 16:29 1888JMX(Java Management Extension ... -
jsf的eclipse开发插件
2008-08-18 12:36 1152大家可以介绍个好的jsf eclipse开发插件吗?要支持js ... -
关于使用hibernate的关联关系还是使用视图的问题
2008-08-02 02:07 1473现在做的工程是HHS框架结构的,在使用hibernate的数据 ... -
java序列化备份及还原
2008-07-29 21:05 1776今天工作遇到一个要把查询结果的List序列化保存到本地机,再上 ...
相关推荐
Spring JdbcTemplate调用Oracle存储过程输出游标结果集实现增删改查
前几天一直在搞spring+hibernate执行存储过程的技术,在网上查了很多资料没有一个是完全的能执行的,代码简单但,几天时间比较辛苦,所以要分多了点。由于包都太大,所以删掉啦!
使用Spring的JdbcTemplate调用Oracle的存储过程
spring mvc + mybatis 调用mysql 存储过程
能不能写个动态的业务,只输入存储过程名称,自动获取存储过程参数,并且参数的数据从前台传递过来,这个就通用了。只写一个通用方法,就可以调用所有的存储过程。只根据输入不同的存储过程名称、参数内容,自动调用...
在已有的spring+mybatis 基本操作oracle数据库的基础上,增加了3个调用存储过程的例子:无返回值、返回结果集、返回多个结果,希望对大家能有所帮助
Spring jdbcTemplate调用Oracle存储过程返回List集合
使用Java struts+spring_ibaits+调用Mysql存储过程 实现增删改查
Spring Bean创建初始化流程
spring运行过程中动态注册bean,代码实例
(1)使用Spring JDBC实现书店的购书过程,即有如下一个BookShopDao接口,编写BookShopDaoImp类实现该接口中的所有方法,并通过JUnit测试这些方法。 (2)(2) 在BookShopDao中添加一个purchase购书方法,其操作流程是...
主要给大家介绍了关于Spring boot调用Oracle存储过程的两种方式及完整代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
IDEA+SpringBoot+SpringSecurity:整个SpringSecurity的实现过程,可应用于线上产品
本篇文章主要介绍了Spring Data JPA调用存储过程实例代码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下。
NULL 博文链接:https://xiaogui9317170.iteye.com/blog/286401
怎么使用MyBatis调用存储过程,步骤详细清楚,一看就会配置
Struts2+Spring+Hibernate整合过程详解 Struts作为MVC 2的Web框架,自推出以来不断受到开发者的追捧,得到广泛的应用。作为最成功的Web框架,Struts拥有众多的优点:MVC 2模型的使用、功能齐全的标志库(Tag ...
beanFactory.preInstantiateSingletons()的初始化过程详解,包含了所有的调用过程。
赠送jar包:spring-cloud-gateway-server-3.1.1....使用方法:解压翻译后的API文档,用浏览器打开“index.html”文件,即可纵览文档内容。 人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译,请放心使用。
Spring 5.2.9 06 源码分析-spring自定义标签解析过程