在使用@SqlResultSetMapping的@ConstructorResult时,我不知道如何在@ColumnResult类型中使用Enum
@SqlResultSetMapping(name="DetailAndResult",
classes={
@ConstructorResult(targetClass=DetailAndResult.class, columns={
@ColumnResult(name="id", type= String.class),
@ColumnResult(name="runId", type=Integer.class),
@ColumnResult(name="subRunId", type=Integer.class),
@ColumnResult(name="transactionId", type=Integer.class),
@ColumnResult(name="referenceNumber", type=String.class),
@ColumnResult(name="customerName", type=String.class),
@ColumnResult(name="transactionType", type=TransactionType.class),
@ColumnResult(name="transactionResultStatus", type=String.class)
})
}
)
在上面的配置中,名称'transactionType'是TransactionType Enum。在这里使用Enum的正确方法是什么。
如果上面是正确的方法,那么我得到这个异常(如果我将删除Enum字段,那么没有异常)所以认为应该有另一种方法来使用它。
Caused by: javax.persistence.PersistenceException: org.hibernate.type.SerializationException: could not deserialize
at org.hibernate.jpa.spi.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1763) ~[hibernate-entitymanager-4.3.6.Final.jar:4.3.6.Final]
at org.hibernate.jpa.spi.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1677) ~[hibernate-entitymanager-4.3.6.Final.jar:4.3.6.Final]
at org.hibernate.jpa.internal.QueryImpl.getResultList(QueryImpl.java:458) ~[hibernate-entitymanager-4.3.6.Final.jar:4.3.6.Final]
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) ~[na:1.7.0_51]
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57) ~[na:1.7.0_51]
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) ~[na:1.7.0_51]
at java.lang.reflect.Method.invoke(Method.java:606) ~[na:1.7.0_51]
at org.springframework.orm.jpa.SharedEntityManagerCreator$DeferredQueryInvocationHandler.invoke(SharedEntityManagerCreator.java:333) ~[spring-orm-4.0.5.RELEASE.jar:4.0.5.RELEASE]
at com.sun.proxy.$Proxy146.getResultList(Unknown Source) ~[na:na]
使用hibernateTemplate,我们使用的是sqlquery.addscalar,并且有一种方法可以使用org.hibernate.type.Type和
来使用Enum。TypeLocatorImpl(new TypeResolver()).custom(EnumType.class, params)
请建议这样的事情是否会用于@SqlResultSetMapping和@ConstructorResult
答案 0 :(得分:9)
我也遇到了这个问题,经过相当广泛的搜索后没有找到任何东西。我已经远远看到了源代码,据我所知,根本没有对枚举进行任何处理(当然我可能会遗漏一些东西)。
我最终做的是创建一个替代构造函数,它为枚举类型接受一个String,然后将它传递给枚举的valueOf()方法。
E.g。
更改@SqlResultSetMapping来执行此操作:
@ColumnResult(name="transactionType", type=String.class),
然后在你班级的构造函数中:
public DetailAndResult(..., String transactionType, ...) {
...
this.transactionType = TransactionType.valueOf(transactionType);
...
}
令人讨厌的是我们必须这样做,但只要您的枚举在数据库中存储为String(即实体上的列使用@Enumerated(EnumType.STRING)注释),它就可以工作。
答案 1 :(得分:0)
遇到了这个问题的无证解决方案。在@ColumnResult
的类型中,您可以放置一个休眠UserType来映射构造函数所期望的类型。这适用于休眠,我不确定其他JPA实现是否支持这一点。
因此,在您的示例中,您将为枚举实现自定义UserType,并将该类放在@ColumnResult
中。
答案 2 :(得分:0)
您仍然可以在实体字段上使用 @Enumerated(EnumType.STRING),
我的@SqlResultSetMapping 看起来有点不同:
@FieldResult(name="state", column="STATE"),