我正在使用旧版Oracle数据库,其中所有存储过程都是使用OUT参数 last 定义的。 Hibernate仅支持存储过程,其中OUT参数定义为 first 。
使用Hibernate ORM 4.3.5,有没有办法覆盖Hibernate的Oracle驱动程序,使用 last 参数而不是 first 来获取值来自@NamedNativeQuery
?我开始使用CallableStatement
的路径,但这违背了使用Hibernate和JPA的全部目的。
以下是我尝试定义@NamedNativeQuery
的方法,如果可能的话,我想坚持这个定义:
@NamedNativeQuery(
name = "MyEntity.storedProc",
query = "call ENTITY_PKG.StoredProc(:inputOne, :inputTwo, :resultSet)",
callable = true,
readOnly = true,
resultClass = MyEntity.class
)
更新:
在跟进Neil Stockton的回答后,我想出了以下@NamedStoredProcedureQuery
注释:
@NamedStoredProcedureQuery(
name = "MyEntity.storedProc",
procedureName = "ENTITY_PKG.StoredProc",
resultClasses = MyEntity.class,
parameters = {
@StoredProcedureParameter(name = "I_INPUT_ONE", type = String.class, mode = ParameterMode.IN),
@StoredProcedureParameter(name = "I_INPUT_TWO", type = Character.class, mode = ParameterMode.IN),
@StoredProcedureParameter(name = "O_CURSOR", type = void.class, mode = ParameterMode.REF_CURSOR)
}
)
但是,使用上述结果会导致此异常... java.lang.UnsupportedOperationException: org.hibernate.dialect.Oracle10gDialect does not support resultsets via stored procedures
。这是堆栈跟踪:
Caused by: java.lang.UnsupportedOperationException: org.hibernate.dialect.Oracle10gDialect does not support resultsets via stored procedures
at org.hibernate.dialect.Dialect.registerResultSetOutParameter(Dialect.java:1612)
at org.hibernate.engine.jdbc.cursor.internal.StandardRefCursorSupport.registerRefCursorParameter(StandardRefCursorSupport.java:94)
at org.hibernate.procedure.internal.AbstractParameterRegistrationImpl.prepare(AbstractParameterRegistrationImpl.java:298)
at org.hibernate.procedure.internal.ProcedureCallImpl.buildOutputs(ProcedureCallImpl.java:417)
at org.hibernate.procedure.internal.ProcedureCallImpl.getOutputs(ProcedureCallImpl.java:378)
at org.hibernate.jpa.internal.StoredProcedureQueryImpl.outputs(StoredProcedureQueryImpl.java:251)
at org.hibernate.jpa.internal.StoredProcedureQueryImpl.execute(StoredProcedureQueryImpl.java:234)
at com.example.MyEntityDao.testStoredProc(MyEntityDao.java:88)
当我尝试执行这样的查询时会发生这种情况:
StoredProcedureQuery query = entityManager
.createNamedStoredProcedureQuery("MyEntity.storedProc");
query.setParameter("I_INPUT_ONE", "inputOneValue");
query.setParameter("I_INPUT_TWO", 'x');
query.execute(); // exception is triggered by query.execute()
MyEntity myEntity = (MyEntity) query.getOutputParameterValue("O_CURSOR");
这本身并没有多大意义,因为Oracle10gDialect
继承了registerResultSetOutParameter()
的实现。此方法为@Override
n Oracle8iDialect
n Oracle9iDialect
,它是Oracle10gDialect
的超类,后者又是<properties>
的超类。
我的Hibernate方言在persistence.xml
的{{1}}部分中定义:
<property name="hibernate.dialect" value="org.hibernate.dialect.Oracle10gDialect" />
作为参考,这是我的存储过程的规范......正文没有RETURN
,因此OUT
光标是我访问结果的唯一方法。
CREATE OR REPLACE PACKAGE ENTITY_PKG IS
TYPE EntityCursor IS REF CURSOR;
PROCEDURE StoredProc
(
I_INPUT_ONE IN ENTITIES.INPUT_ONE%TYPE, -- VARCHAR2
I_INPUT_TWO IN ENTITIES.INPUT_TWO%TYPE, -- CHAR
O_CURSOR OUT EntityCursor
);
END ENTITY_PKG;
我做错了什么?
答案 0 :(得分:1)
JPA 2.1提供了一个API来使用存储过程,而不是将它们作为本机“查询”进行黑客攻击。也许你真的应该用它?