我正在使用 JDev 11.1.1.7.0 和 Oracle Exp 11g DB
在我的 ADF 应用程序中,我试图调用一个存储过程,但我不断收到错误,最后一个是“无效的列类型:1111”。
我认为问题出在这条线上
st.registerOutParameter(8,Types.OTHER);
我的AppModuleImpl完整方法:
public Row callProcWithRowOut(Object[] bindVars) {
CallableStatement st = null;
try {
st = getDBTransaction().createCallableStatement("begin ADF_ITEM_SOLD(?,?,?,?,?,?,?,?);end;",getDBTransaction().DEFAULT);
st.registerOutParameter(8,Types.OTHER);
if (bindVars != null) {
for (int z = 0; z < bindVars.length; z++) {
st.setObject(z + 1, bindVars[z]);
}
}
st.executeUpdate();
return (Row)st.getObject(8);
} catch (SQLException e) {
throw new JboException(e);
} finally {
if (st != null) {
try {
st.close();
}
catch (SQLException e) {}
}
}
}
痕迹:
java.sql.SQLException: Invalid column type: 1111
at oracle.jdbc.driver.OracleStatement.getInternalType(OracleStatement.java:5344)
at oracle.jdbc.driver.OracleCallableStatement.registerOutParameterInternal(OracleCallableStatement.java:153)
at oracle.jdbc.driver.OracleCallableStatement.registerOutParameter(OracleCallableStatement.java:399)
at oracle.jdbc.driver.OracleCallableStatement.registerOutParameter(OracleCallableStatement.java:581)
at oracle.jdbc.driver.OracleCallableStatementWrapper.registerOutParameter(OracleCallableStatementWrapper.java:1765)
at weblogic.jdbc.wrapper.CallableStatement_oracle_jdbc_driver_OracleCallableStatementWrapper.registerOutParameter(Unknown Source)
at model.BC.AppModule.AppModule_AMImpl.callProcWithRowOut(AppModule_AMImpl.java:1793)
at model.BC.Views.SalesInvoiceItems_VOImpl.insertItems(SalesInvoiceItems_VOImpl.java:54)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at oracle.adf.model.binding.DCInvokeMethod.invokeMethod(DCInvokeMethod.java:657)
at oracle.adf.model.binding.DCDataControl.invokeMethod(DCDataControl.java:2143)
at oracle.adf.model.bc4j.DCJboDataControl.invokeMethod(DCJboDataControl.java:3118)
at oracle.adf.model.binding.DCInvokeMethod.callMethod(DCInvokeMethod.java:261)
at oracle.jbo.uicli.binding.JUCtrlActionBinding.doIt(JUCtrlActionBinding.java:1635)
at oracle.adf.model.binding.DCDataControl.invokeOperation(DCDataControl.java:2150)
at oracle.jbo.uicli.binding.JUCtrlActionBinding.invoke(JUCtrlActionBinding.java:740)
at oracle.adf.controller.v2.lifecycle.PageLifecycleImpl.executeEvent(PageLifecycleImpl.java:407)
at oracle.adfinternal.view.faces.model.binding.FacesCtrlActionBinding._execute(FacesCtrlActionBinding.java:252)
at oracle.adfinternal.view.faces.model.binding.FacesCtrlActionBinding.execute(FacesCtrlActionBinding.java:210)
at view.backing.SalesInvoicesUpdate.insertItems_action(SalesInvoicesUpdate.java:281)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at com.sun.el.parser.AstValue.invoke(Unknown Source)
at com.sun.el.MethodExpressionImpl.invoke(Unknown Source)
at org.apache.myfaces.trinidad.component.MethodExpressionMethodBinding.invoke(MethodExpressionMethodBinding.java:46)
at com.sun.faces.application.ActionListenerImpl.processAction(ActionListenerImpl.java:102)
at org.apache.myfaces.trinidad.component.UIXCommand.broadcast(UIXCommand.java:190)
可能是什么问题?
谢谢你的时间。
正如我在评论中所说,通常不可能调用返回
non-sql
类型的存储过程(使用特定于数据库的 API 有一些例外)。由于 %ROWTYPE
是 Pl/Sql
类型,您将必须修改您的过程以返回支持的(即 SQL
)类型。最简单的方法是返回一些与您的 %ROWTYPE
相对应的 OUT 参数,或者您甚至可以返回一个简单的字符串/varchar(该字符串可能包含所有属性的逗号分隔列表),然后解析该字符串客户端(Java)端。
我还在此处编写了一个可用的简单示例,展示了如何处理这种情况的一种方法 (http://easyorm.info/SProc)。此示例使用 EasyORM 库(简单的
JDBC
包装器),而不是普通的 JDBC,但它应该让您了解如何修改存储过程。
您还可以将
%ROWTYPE
编码为 Oracle 对象(sql
类型)并让存储过程返回该对象,但这种方法更复杂(请参阅Mapping an Oracle stored procedure result to a custom Java type (class) 举个例子)
同样的情况,但我无法修改存储过程..还有其他解决方案吗?我怎么能用 %rowtype 参数调用存储过程?