5

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

由于 id 是由数据库自动生成的,因此在 em.persist(myEntity) 调用之前它为空。但是在持久化之后,它有一个由 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() 方法期间被标记为脏,并且在事务结束时刷新期间会发生更新。但显然不是。

我使用休眠 v4.2.4

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

4

5 回答 5

2

而已!我进行了额外的测试,实际上,在刷新期间调用了@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 被赋值,实体脏。然后不再执行刷新(以及脏检查和更新)=> 不保存数据库。非常感谢!!!

于 2013-08-21T07:19:49.103 回答
0

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

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

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

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

于 2013-08-21T03:01:02.223 回答
0

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

但是,如果您在数据层中确实需要它,请使用 @PrePersist。

于 2013-10-16T06:37:53.967 回答
0

利用

@PrePersist

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

有关如何处理业务 ID 的更多示例,我建议您查看有关 equals()/hashcode() 困境的线程,特别是此响应:https ://stackoverflow.com/a/5103360/2598693

于 2013-08-21T07:41:55.543 回答
0

您正在执行 POST 持久化操作,因此它发生在写入数据库之后。

根据对JPA @PostPersist 用法的回答,规范指出

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

于 2013-08-20T15:09:40.703 回答