hibernate级联保存问题,出错not-null property references a null or transient value:

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了hibernate级联保存问题,出错not-null property references a null or transient value:相关的知识,希望对你有一定的参考价值。

问题大致是这样的,实体A和B双向一对一映射,
A a = new A();
B b = new B();
a.setB(b);
b.setA(a);

两边都是many-to-one,设置cascade=save-update,not-null=true

然后dao.save(a);就出错。

hibernate进行级联处理时,关于一对一的关系应该注意的问题:

1、在进行hibernate注解配置的时候,他们的关系应该如下:

2、entity  aword

3、有了以上的hibernate级联关系后,删除奖品项,不会影响到奖品,那么也不会影响到其他使用了这个奖品的奖品项。这里还需要注意几点:

1)在数据库中,需要在奖品项item中添加外键字段awordId,因此可以通过这个奖品id取出对应的奖品。

2)在做奖品aword删除时,切记要使用hibernate的delete删除方法,如果删除多个奖品,就用for循环删除。不能使用hibernate调用query方法,不能去使用sql删除数据,否则会绕过hibernate机制,不会进行级联删除等操作。有些类似于spring的session,如果你自己openSession,那么这个session就需要自己打开关闭,并且还需要放在事务中处理。但是如果使用getCurrentSession,那么spring会帮你管理这个session,前提你需要在spring配置文件中配置事务。

4、hibernate如果使用延迟加载机制,如果使用调试,那么调试的数据是看不到的,需要打印,如奖品项item,如果使用延迟加载,调试是看不到item的数据的,如果想看item的某一个数据,可以打印这个数据。

hibernate延迟加载只会将数据保存在session中(未确认),那么如果你在dao层去数据,那么在service层使用数据时,数据可能报错,原因时数据取出来后session就会关闭,那么其他层,或者web就不能得到数据。解决这个问题的方法是,让延迟加载的范围扩大到一次请求,可以在web.xml中配置监听器进行处理。

参考技术A not-null=false 就不会报错了,插入的是个空值本回答被提问者和网友采纳 参考技术B 出错信息是说,一个要求不能为空的属性为空值(null),你仔细检查下保存的数据,里面应该有要求非空的属性没有赋值。

如何在 Hibernate 3.6 中正确级联保存主键上的一对一双向关系

【中文标题】如何在 Hibernate 3.6 中正确级联保存主键上的一对一双向关系【英文标题】:How do I properly cascade save a one-to-one, bidirectional relationship on primary key in Hibernate 3.6 【发布时间】:2011-04-30 23:14:45 【问题描述】:

我与共享密钥建立了一对一的双向实体关系。当我尝试保存关联的所有者时,我得到一个针对关系拥有方的“生成空 id”异常。我正在使用 hibernate-entitymanager 并使用 spring 进行事务管理。

拥有实体

@Entity
@Table(name = "lead")
public class Lead

    private Long leadId;

    private LeadAffiliate leadAffiliate;

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    public Long getLeadId()
    
        return leadId;
    

    @OneToOne(cascade = CascadeType.ALL)
    @PrimaryKeyJoinColumn
    public LeadAffiliate getLeadAffiliate()
    
        return leadAffiliate;
    

自有实体

@Entity
@Table(name = "lead_affiliate")
public class LeadAffiliate

    private Long leadId;

    private Lead lead;

    @Id
    public Long getLeadId()
    
        return leadId;
    

    @MapsIdmappedBy = "leadAffiliate")
    @OneToOne(cascade = CascadeType.All)
    @PrimaryKeyJoinColumn
    @JoinColumn(name = "lead_id")
    public Lead getLead()
    
        return lead;
    

下面的代码用于保存实体:

LeadAffiliate aff = new LeadAffiliate();

aff.setLead(lead);
lead.setLeadAffiliate(aff);

em.persist(lead);

这一切在休眠 3.5.0-Final 中都可以正常工作。当尝试升级到 3.5.6-Final 或 3.6.0.Final 时,我开始收到“为 LeadAffiliate 生成空 id”错误:

javax.persistence.PersistenceException: org.hibernate.id.IdentifierGenerationException: null id generated for:class com.sellingsource.bizdev.entities.LeadAffiliate
    at org.hibernate.ejb.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1214)
    at org.hibernate.ejb.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1147)
    at org.hibernate.ejb.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1153)
    at org.hibernate.ejb.AbstractEntityManagerImpl.persist(AbstractEntityManagerImpl.java:678)
    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 org.springframework.orm.jpa.ExtendedEntityManagerCreator$ExtendedEntityManagerInvocationHandler.invoke(ExtendedEntityManagerCreator.java:365)
    at $Proxy152.persist(Unknown Source)
    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 org.springframework.orm.jpa.SharedEntityManagerCreator$SharedEntityManagerInvocationHandler.invoke(SharedEntityManagerCreator.java:240)
    at $Proxy120.persist(Unknown Source)
    at com.sellingsource.common.dao.JpaGenericDao.create(JpaGenericDao.java:38)
    ... 64 more
Caused by: org.hibernate.id.IdentifierGenerationException: null id generated for:class com.sellingsource.bizdev.entities.LeadAffiliate
    at org.hibernate.event.def.AbstractSaveEventListener.saveWithGeneratedId(AbstractSaveEventListener.java:123)
    at org.hibernate.ejb.event.EJB3PersistEventListener.saveWithGeneratedId(EJB3PersistEventListener.java:69)
    at org.hibernate.event.def.DefaultPersistEventListener.entityIsTransient(DefaultPersistEventListener.java:179)
    at org.hibernate.event.def.DefaultPersistEventListener.onPersist(DefaultPersistEventListener.java:135)
    at org.hibernate.impl.SessionImpl.firePersist(SessionImpl.java:799)
    at org.hibernate.impl.SessionImpl.persist(SessionImpl.java:791)
    at org.hibernate.engine.EJB3CascadingAction$1.cascade(EJB3CascadingAction.java:48)
    at org.hibernate.engine.Cascade.cascadeToOne(Cascade.java:392)
    at org.hibernate.engine.Cascade.cascadeAssociation(Cascade.java:335)
    at org.hibernate.engine.Cascade.cascadeProperty(Cascade.java:204)
    at org.hibernate.engine.Cascade.cascade(Cascade.java:161)
    at org.hibernate.event.def.AbstractSaveEventListener.cascadeBeforeSave(AbstractSaveEventListener.java:450)
    at org.hibernate.event.def.AbstractSaveEventListener.performSaveOrReplicate(AbstractSaveEventListener.java:282)
    at org.hibernate.event.def.AbstractSaveEventListener.performSave(AbstractSaveEventListener.java:203)
    at org.hibernate.event.def.AbstractSaveEventListener.saveWithGeneratedId(AbstractSaveEventListener.java:129)
    at org.hibernate.ejb.event.EJB3PersistEventListener.saveWithGeneratedId(EJB3PersistEventListener.java:69)
    at org.hibernate.event.def.DefaultPersistEventListener.entityIsTransient(DefaultPersistEventListener.java:179)
    at org.hibernate.event.def.DefaultPersistEventListener.onPersist(DefaultPersistEventListener.java:135)
    at org.hibernate.event.def.DefaultPersistEventListener.onPersist(DefaultPersistEventListener.java:61)
    at org.hibernate.impl.SessionImpl.firePersist(SessionImpl.java:808)
    at org.hibernate.impl.SessionImpl.persist(SessionImpl.java:782)
    at org.hibernate.impl.SessionImpl.persist(SessionImpl.java:786)
    at org.hibernate.ejb.AbstractEntityManagerImpl.persist(AbstractEntityManagerImpl.java:672)
    ... 77 more

顺便说一句,我不确定 Lead Affiliate 上的注释一开始就完全正确。他们工作,但似乎有点笨拙。所以我把它们改成了:

@Entity
@Table(name = "lead_affiliate")
public class LeadAffiliate

    private Long leadId;

    private Lead lead;

    @Id
    @GenericGenerator(name = "foreign", strategy = "foreign", parameters = 
                    @org.hibernate.annotations.Parameter(name = "property", value="lead")
    )
    @GeneratedValue(generator = "foreign")
    public Long getLeadId()
    
        return leadId;
    

    @OneToOne(mappedBy = "leadAffiliate")
    @PrimaryKeyJoinColumn
    public Lead getLead()
    
        return lead;
    

但是,通过这些更改,我得到了相同的结果。 (适用于 3.5.0,但不适用于 3.5.6 或 3.6.0)

我是否需要一种新的方法来执行此操作,或者这是一个错误?我担心的是我的代码由于错误而目前正在运行:/。

【问题讨论】:

我使用普通的 JPA 1.0 来实现你的目标请参阅 ***.com/questions/2001007/2039553#2039553 只需将 Hibernate Cascade 注释替换为其 JPA 对应项 【参考方案1】:

规范说派生实体应该是关系的拥有方:

2.4.1 派生身份对应的主键

一个人的身份 实体可能来源于 另一个实体的身份( “父”实体)当前者 实体(“依赖”实体)是 多对一或一对一的所有者 与母公司的关系和 外键映射关系 从受抚养人到父母。

在您的情况下,LeadAffiliate 是派生的,所以它应该是所有者,而 Lead 应该被 mappedBy 标记为非拥有方。以下适用于 3.5.0 和 3.5.6:

public class Lead  
    @Id @GeneratedValue
    private Long leadId; 
 
    @OneToOne(cascade = CascadeType.ALL, mappedBy = "lead")
    private LeadAffiliate leadAffiliate; 

    ...

.

public class LeadAffiliate   
    @Id
    private Long leadId;  
  
    @OneToOne @MapsId
    private Lead lead; 

    ...

【讨论】:

不幸的是,Hibernate 不喜欢@Id 方法(需要填补一个新的错误......)。所以 +1 是一个适用于 Hibernate 的解决方案。 我已确认此更改适用于 3.5.6 我确实必须添加一个 @JoinColumn 注释,因为它试图使用错误的列名。但在那之后它运作良好。我还验证了它在 3.6.0 中有效。听起来确实像,根据文档,这是休眠 3.5 中的一个缺陷,它允许我的原始注释工作。 在这个问题上卡了 2 天,这个答案肯定会得到更多的支持【参考方案2】:

我的回答不会解释为什么 Hibernate 3.5.0-Final 可以正常工作,但 3.5.6-Final 或 3.6.0.Final 不能正常工作(你应该报告这个,我称之为回归)。

无论如何,JPA 2.0 以标准方式更好地支持派生标识符,在您的情况下,我认为您可以简单地使用 Id 注释来注释您的 OneToOne 关系。

更新:正如 axtavt 强调的那样,当使用派生标识符时,“依赖”实体必须是关系的所有者。所以依赖实体的完整映射将是:

@Entity
@Table(name = "lead_affiliate")
public class LeadAffiliate 
    private Lead lead;

    @Id
    @OneToOne
    @JoinColumn(name="FK")
    public Lead getLead() 
        return lead;
    

以及“父”实体:

@Entity
@Table(name = "lead")
public class Lead 
    private Long leadId;

    private LeadAffiliate leadAffiliate;

    @Id @GeneratedValue(strategy = GenerationType.AUTO)
    public Long getLeadId() 
        return leadId;
    

    @OneToOne(cascade = CascadeType.ALL, mappedBy="lead")
    public LeadAffiliate getLeadAffiliate() 
        return leadAffiliate;
    

这是一个有效的 JPA 2.0 映射,可与 EclipseLink 一起使用。然而,Hibernate 不喜欢它并且不会实例化 EntityManagerFactory(该死!)。

作为解决方法,您必须使用solution suggested by axtavt 即声明主键属性以及关系属性,并在关系属性上使用MapsId

但是上面应该可以工作,IMO 在 Hibernate 中有一个错误(报告为HHH-5695)。

参考文献

JPA 2.0 规范 第 2.4.1 节“派生身份对应的主键”(冗长,涵盖很多情况) JPA 维基书 Primary Keys through OneToOne Relationships

【讨论】:

我使用 OpenJPA 2.2.2,这对我有用(JPA 2.0)

以上是关于hibernate级联保存问题,出错not-null property references a null or transient value:的主要内容,如果未能解决你的问题,请参考以下文章

OneToMany Hibernate 保存级联问题

级联保存不适用于 Hibernate 4 和 @OneToMany

Hibernate:OneToMany 通过级联保存孩子

hibernate---级联保存级联删除

Hibernate--cascade级联

org.hibernate.PropertyValueException: not-null property references a null or transient value