我有一个分层实体,它将自己作为父级引用。我只需要通过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。
有什么想法吗?
另外请不要建议我使用延迟加载。在我的情况下这是不可行的(再次,太难解释)。
答案 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;