【问题标题】:Field modified in @PostPersist is not updated in the DB@PostPersist 中修改的字段未在数据库中更新
【发布时间】:2013-08-22 17:06:01
【问题描述】:

MyEntity 是具有生成 id 的经典 JPA 实体。我想根据 id 自动将一个字段(myStringField)设置为一个值。在本例中,如果 id 为 43,我想将其设置为“foo43”。

由于 id 是由 DB 自动生成的,因此在 em.persist(myEntity) 调用之前它为 null。但是在持久化之后,它有一个由 Hibernate 分配的 id(在 DB 序列上称为 NEXTVAL)。所以,我想声明一个@PostPersist 方法,如下:

package ...;

import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.PostPersist;


@Entity
public class MyEntity {

    @Id   @GeneratedValue(strategy = GenerationType.AUTO)
    private Long id;

    private String myStringField;

    @PostPersist
    public void postPersist(){
        System.out.println("The id = " + id);
        this.myStringField = "foo" + id;
    }

    public String getMyStringField() {
        return myStringField;
    }
    ...
}

下面是一些创建和持久化实体的代码:

package ...;

import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;

import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

@Transactional
@Service
public class MySpringBean {
   @PersistenceContext EntityManager em;

   public void createMyEntity() {
       MyEntity myEntity = new MyEntity();
       em.persist(myEntity);
       System.out.println(myEntity.getMyStringField());
   }
}

它打印以下输出:

The id = 47
foo47

这很好。但是,在数据库中,字段 myStringField 是 null !有人知道为什么吗?

我确实希望实体在 MyEntity.postPersist() 方法期间被标记为脏,并且在事务结束时,刷新期间会发生更新。但显然不是。

我使用 Hibernate v4.2.4

有人知道为什么这个数据库记录有一个空值吗?

【问题讨论】:

    标签: hibernate jpa


    【解决方案1】:

    就是这样!我进行了额外的测试,实际上,在刷新期间调用了@PostPersist(可能是因为在刷新期间发送了 SQL INSERT)。因此,总而言之,事情按以下顺序发生:1. em.persist(),1.1 NEXTVAL 发送到 DB 并分配 myEntity.id,2 @Transactional 方法结束并执行刷新。 2.1 INSERT 发送到数据库。 2.2 @PostPersist 调用。 2.2.1 myEntity.myStringField 被赋值,实体脏。然后不再执行刷新(以及脏检查和更新)=> 不保存数据库。非常感谢!!!

    【讨论】:

      【解决方案2】:

      你正在做 POST 持久化,所以它发生在写入数据库之后。

      根据对JPA @PostPersist usage 的回答,规范声明

      These database operations may occur directly after the persist, merge, or remove operations have been invoked or they may occur directly after a flush operation has occurred

      【讨论】:

        【解决方案3】:

        @PostPersist 代码在 MyEntity 保存后运行。因此,如果您想确保字段 myStringField 使用更新后的值保存:“foo”+ id,那么在 @PostPersist 代码之后,您将需要再次保存实体。

        因此,如果您将代码更改为:

           MyEntity myEntity = new MyEntity();
           em.persist(myEntity);
           em.persist(myEntity);
        

        这个,理论上应该正确保存更新的 myStringField。不过这有点尴尬。不确定您的整个用例,但您可以考虑在数据库本身中编写一个“插入触发器”来为您解决这个问题。

        【讨论】:

          【解决方案4】:

          使用

          @PrePersist
          

          相反,但是您的方法字段生成对我来说不正确,因为如果您的 myStringField 即将成为业务 id,与 jpa id 不同,您应该假设它是在 bean 初始化时创建的。

          有关如何处理业务 id 的更多示例,我建议您查看有关 equals()/hashcode() 困境的帖子,特别是此回复:https://stackoverflow.com/a/5103360/2598693

          【讨论】:

            【解决方案5】:

            要回答您的问题,您不应该在数据层中做那种事情。该插入和更新应由您的业务逻辑层处理。

            但是如果你真的需要在数据层使用@PrePersist。

            【讨论】:

              猜你喜欢
              • 1970-01-01
              • 1970-01-01
              • 1970-01-01
              • 1970-01-01
              • 1970-01-01
              • 1970-01-01
              • 2020-06-06
              • 1970-01-01
              • 1970-01-01
              相关资源
              最近更新 更多