Hibernate:没有实体类的外键,只有id

时间:2015-09-22 17:14:21

标签: java entity-framework hibernate foreign-keys

我有一个分层实体,它将自己作为父级引用。我只需要通过id进行映射,而不是通过实体实例进行映射(原因太复杂,无法解释)。所以我用这种方式定义了实体:

class Item {

    @Id
    private String id;

    @ManyToOne(targetEntity = Item.class)
    @JoinColumn(name = "PARENT_ID", nullable = true)
    private String parentId;

}

这似乎工作正常。在数据库中正确创建外键约束。但是当我执行以下查询时:

SELECT i FROM Item i WHERE i.parentId = :parentId

我得到了这个例外(有趣的部分是粗体):

org.hibernate.PropertyAccessException:调用com.example.dom.Item.id的getter时出现IllegalArgumentException     在org.hibernate.property.BasicPropertyAccessor $ BasicGetter.get(BasicPropertyAccessor.java:192)     at org.hibernate.tuple.entity.AbstractEntityTuplizer.getIdentifier(AbstractEntityTuplizer.java:346)     在org.hibernate.persister.entity.AbstractEntityPersister.getIdentifier(AbstractEntityPersister.java:4746)     在org.hibernate.persister.entity.AbstractEntityPersister.isTransient(AbstractEntityPersister.java:4465)     在org.hibernate.engine.internal.ForeignKeys.isTransient(ForeignKeys.java:243)     at org.hibernate.engine.internal.ForeignKeys.getEntityIdentifierIfNotUnsaved(ForeignKeys.java:293)     在org.hibernate.type.EntityType.getIdentifier(EntityType.java:537)     在org.hibernate.type.ManyToOneType.nullSafeSet(ManyToOneType.java:174)     在org.hibernate.param.NamedParameterSpecification.bind(NamedParameterSpecification.java:67)     在org.hibernate.loader.hql.QueryLoader.bindParameterValues(QueryLoader.java:616)     在org.hibernate.loader.Loader.prepareQueryStatement(Loader.java:1901)     在org.hibernate.loader.Loader.executeQueryStatement(Loader.java:1862)     在org.hibernate.loader.Loader.executeQueryStatement(Loader.java:1839)     在org.hibernate.loader.Loader.doQuery(Loader.java:910)     在org.hibernate.loader.Loader.doQueryAndInitializeNonLazyCollections(Loader.java:355)     在org.hibernate.loader.Loader.doList(Loader.java:2554)     在org.hibernate.loader.Loader.doList(Loader.java:2540)     在org.hibernate.loader.Loader.listIgnoreQueryCache(Loader.java:2370)     在org.hibernate.loader.Loader.list(Loader.java:2365)     在org.hibernate.loader.hql.QueryLoader.list(QueryLoader.java:497)     在org.hibernate.hql.internal.ast.QueryTranslatorImpl.list(QueryTranslatorImpl.java:387)     在org.hibernate.engine.query.spi.HQLQueryPlan.performList(HQLQueryPlan.java:236)     在org.hibernate.internal.SessionImpl.list(SessionImpl.java:1300)     在org.hibernate.internal.QueryImpl.list(QueryImpl.java:103)     在com.example.dao.ItemDao.findChildrenByParentId(ItemDao.java:43)     在com.example.dao.ItemDao $$ FastClassBySpringCGLIB $$ 51b04ce9.invoke()     在org.springframework.cglib.proxy.MethodProxy.invoke(MethodProxy.java:204)     在org.springframework.aop.framework.CglibAopProxy $ CglibMethodInvocation.invokeJoinpoint(CglibAopProxy.java:717)     在org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:157)     at org.springframework.dao.support.PersistenceExceptionTranslationInterceptor.invoke(PersistenceExceptionTranslationInterceptor.java:136)     ......还有47个 引起:java.lang.IllegalArgumentException:object不是声明类的实例     at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)     at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)     at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)     在java.lang.reflect.Method.invoke(Method.java:597)     在org.hibernate.property.BasicPropertyAccessor $ BasicGetter.get(BasicPropertyAccessor.java:169)     ......还有76个

似乎Hibernate正在尝试使用属性parentId,就好像它是Item类型,而不是String。

有什么想法吗?

另外请不要建议我使用延迟加载。在我的情况下这是不可行的(再次,太难解释)。

3 个答案:

答案 0 :(得分:1)

关联使用实体引用(在这种情况下需要你使用真实对象Item)如果你想使用普通ID列,那么你说你不希望hibernate管理它们,只是删除关联注释。

答案 1 :(得分:0)

尝试定义id列,例如试试这个。

@Column(name="id")

答案 2 :(得分:0)

据我了解你的问题,这可能有用。

@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "id")
private String id;


@ManyToOne
@JoinColumn(name = "parent_id",nullable = true)
private Item item;