Spring StoredProcedure调用ORACLE存储过程或函数

Spring StoredProcedure调用ORACLE存储过程或函数废话不说,直接上代码:应用实例//引用方式package.function或package.procedureStoredProceduresp=neworg.springframework.jdbc.object.StoredProcedure(ds,”PACKAGE_NAME.PROCEDURE_NAME”);//调用函数时必须,调用存储过程不要sp.setFun

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

废话不说,直接上代码:

应用实例

//引用方式package.function或package.procedure
StoredProcedure sp = new org.springframework.jdbc.object.StoredProcedure(ds, "PACKAGE_NAME.PROCEDURE_NAME");
//调用函数时必须,调用存储过程不要
sp.setFunction(true);
//设置返回参数名(将来通过此名称获取输出的返回结果),返回参数需在IN参数前定义
//返回类型Types.ARRAY,自定义类型名称(自定义oracle的TYPE必须在package之上定义,可以在schema层次定义,package内不支持,参见下段“无效的名称模式”)
//sp.declareParameter(new SqlOutParameter("result", Types.INTEGER)); //输出INTEGER类型
sp.declareParameter(new SqlOutParameter("result", Types.ARRAY, "NUMBER_ARRAY")); //返回table类型

//设置输入参数
sp.addParameter("p_name", "广州");
//定义返回对象
Map<String, Object> result = null;
//执行。存在ORA04068异常的可能,若发生则再试执行一次。
try{
	result = sp.execute();
}catch(UncategorizedSQLException e){				
	String state = e.getSQLException().getSQLState();
	int code = e.getSQLException().getErrorCode();
	if("72000".equals(state) && code == 4068){ //ORA-04068 detected
		result = sp.execute();
	}else{
		throw e;
	}
}finally{
	
}

//强制转换返回结果,对应oracle.sql.ARRAY
//Integer returnCode = (Integer) result.get("result");
ARRAY r = (ARRAY)result.get("result");
//结果处理
long[] la = null;
try{
	la = r.getLongArray();   //简单获取,复杂的使用Datum获取
	if(la!=null && la.length>0){
		System.out.println(la[0]);
	}else{
		System.out.println("no data");
	}
}catch(Exception e){
	System.out.println("ERROR");
}

关于包失效

当前session处在活动状态,如果此时ORACLE的package在另外一个session中被重新编译,那么当前session再执行其中的procedure就会报包失效,异常STACK见下。

报错后oracle会自动更新此session中的包状态,所以再次执行则会成功,如前文代码所示。

org.springframework.jdbc.UncategorizedSQLException: CallableStatementCallback; uncategorized SQLException for SQL [{? = call PACKAGE_NAME.FUNCTION_NAME(?)}]; SQL state [72000]; error code [4068]; ORA-04068: 已丢弃程序包  的当前状态
ORA-04061:  的当前状态失效
ORA-04061: package body "SCHEMA_NAME.PACKAGE_NAME" 的当前状态失效
ORA-04065: 未执行, 已变更或删除 package body "SCHEMA_NAME.PACKAGE_NAME"
ORA-06508: PL/SQL: 无法找到正在调用 : "SCHEMA_NAME.PACKAGE_NAME" 的程序单元
ORA-06512: 在 "SCHEMA_NAME.PACKAGE_NAME", line 432
ORA-06512: 在 "SCHEMA_NAME.PACKAGE_NAME", line 192
ORA-06512: 在 line 1
; nested exception is java.sql.SQLException: ORA-04068: 已丢弃程序包  的当前状态
ORA-04061:  的当前状态失效
ORA-04061: package body "SCHEMA_NAME.PACKAGE_NAME" 的当前状态失效
ORA-04065: 未执行, 已变更或删除 package body "SCHEMA_NAME.PACKAGE_NAME"
ORA-06508: PL/SQL: 无法找到正在调用 : "SCHEMA_NAME.PACKAGE_NAME" 的程序单元
ORA-06512: 在 "SCHEMA_NAME.PACKAGE_NAME", line 432
ORA-06512: 在 "SCHEMA_NAME.PACKAGE_NAME", line 192
ORA-06512: 在 line 1
	at org.springframework.jdbc.support.SQLStateSQLExceptionTranslator.translate(SQLStateSQLExceptionTranslator.java:124)
	at org.springframework.jdbc.support.SQLErrorCodeSQLExceptionTranslator.translate(SQLErrorCodeSQLExceptionTranslator.java:322)
	at org.springframework.jdbc.core.JdbcTemplate.execute(JdbcTemplate.java:952)
	at org.springframework.jdbc.core.JdbcTemplate.call(JdbcTemplate.java:985)
	at org.springframework.jdbc.object.StoredProcedure.execute(StoredProcedure.java:117)

无效的名称模式

当package中使用了自定义array类型时,jdbc调用会出现”无效的名称模式”错误提示。

原因摘自网络:

To my knowledge, the Oracle JDBC driver does not support using the ArrayDescriptor for array data types (varray or nested table) that are defined inside of a package. The same is true for StructDescriptor as well. If you want to use array and object data types, you must define them outside of a package. Then you’ll be able to use the descriptors in your JDBC programs.

As far as I know, you can only create an “ArrayDescriptor” and a “StructDescriptor” for database types.
In other words, types that were created using the CREATE TYPE (DDL) statement.
You cannot create an “ArrayDescriptor” or a “StructDescriptor” for types created in PL/SQL packages

oracle jdbc代码中

ArrayDescriptor.createDescriptor(type, conn);

其中的type只是单独的名字,默认前面是schema,但是不能加package。只有schema级别对象类型是可以的。
解决的办法供参考(1)尝试对package.type建立一个同义词了(2)定义schema级别的对象(自定义类型放在包外)

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/164517.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • VS2015序列号_Idm序列号无效了

    VS2015序列号_Idm序列号无效了1、VS2008简体中文正式版序列号1.VisualStudio2008ProfessionalEdition:XMQ2Y-4T3V6-XJ48Y-D3K2V-6C4WT2.VisualStudio2008TeamTestLoadAgent:WPX3J-BXC3W-BPYWP-PJ8CM-F7M8T3.VisualStudio2008TeamSyst…

  • 室内定位指纹算法_指纹识别算法

    室内定位指纹算法_指纹识别算法一、概述最近在做一个基于蓝牙的室内定位的项目,做了一个三角定位算法,由于室内的环境比较复杂,信号反射折射比较多,很多时候信号的大小(RSSI)跟距离并不是完全一一对应的,可能远的地方信号反而更强,三角质心定位算法就有点不合适了,因此想试用指纹定位算法,看一下指纹定位算法的效果。在此总结一下指纹定位算法。 二、指纹定位算法介绍指纹定位算法是基于室内环境复杂,信号反射折射所形成的在不同

    2022年10月21日
  • phpstrom2020激活码(破解版激活)

    phpstrom2020激活码(破解版激活),https://javaforall.cn/100143.html。详细ieda激活码不妨到全栈程序员必看教程网一起来了解一下吧!

  • 这2个PDF转Word免费不限页数工具很多人没用过

    这2个PDF转Word免费不限页数工具很多人没用过很多人在搜索下载过PDF转换器的小伙伴都会有一个灵魂拷问:难道就没有免费还没页数限制的PDF转Word的工具吗?小编经过不断的对比和试用,找到以下两款好用免费的工具,相信总有一个你能用上。一、PDF转换器相信了解PDF这种文档格式设计由来的人对于Adobe肯定不陌生,所以首先要说的PDF转换工具就是AdobePDF,下载安装后打开软件,直接将PDF拖到软件页面打开即可,然后点击左上角“文件”中的“另存为其他”,选择我们需要转换成的Word格式就可以了。或者点击右侧“工具”选项中的“将文件导出为”并

  • C语言中的strstr函数的用法「建议收藏」

    C语言中的strstr函数的用法「建议收藏」strstr(str1,str2)函数用于判断字符串str2是否是str1的子串。如果是,则该函数返回str2在str1中首次出现的地址;否则,返回NULL。PHP语言函数编辑strstr()函数搜索一个字符串在另一个字符串中的第一次出现。该函数返回字符串的其余部分(从匹配点)。如果未找到所搜索的字符串,则返回false。语法

  • 软件封装工具(牛人自制升降工具)

    平时Java项目的开发通常需要统一管理日志(Log)的输出,例如控制日志信息输送的目的地(控制台、文件等),控制每一条日志的输出格式,把日志分为不同的级别等。常用的比较成熟的Java日志管理工具有Apache的Log4j等。但有时我们平时一时兴趣想写个小Dmeo或小工具,想较好的控制日志的输出,引入专业的日志管理库又显得比较繁琐,下面我就自己封装一个简单的日志工具类(LogUtils.java)。

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号