java - @PreUpdate 更新时不保存父对象

标签 java spring hibernate jpa spring-data

我有两个具有一对多关系的实体。 Parent 可以有多个 Child 实体实例。 我向父项添加了一个字段,用于存储子项修改的日期 (childrenLastModifiedDate)。为了保持这一点,我添加了方法:

@PrePersist
@PreUpdate
@PreRemove
private void handle() {
    parent.setChildrenLastModifiedDate(now());
}

问题来了。它并不总是在 child 被保存时调用。 在本地(mac os),它按预期工作,所有三种类型的更改都被跟踪并保存到父实体。然而,在服务器(linux)上它只适用于:

  • @PrePersist
  • @PreRemove

即使调用 PreUpdate,更改也不会保存。我什至尝试添加直接存储库调用来保存父级。结果是一样的。更新时不保存任何内容,但在删除或保留时保存。 我试图在附加事务中进行此更改并且它起作用了,但它太耗费资源了。另外,我可以看到有人有非常相似的经历:

JPA/Hibernate preUpdate doesn't update parent object

但是,没有关于如何处理问题本身的任何内容。 问题是:有没有办法保证使用此注释将始终有效并执行依赖实体的其他更新?如果不是,处理这种逻辑的最佳方法是什么?对 child 的每次更改都需要更新。就算级联保存。

最佳答案

我遇到了同样的问题。我使用 Hibernate 拦截器修复了它。

首先,您必须在基本实体类中声明一些基本方法,使我们能够找到实体的父级,一种链接更新所需字段的方法。

public abstract class BaseEntity {
    public Optional<BaseEntity> getParent() {
        // subclass should override this method to return the parent entity
        return Optional.empty();
    }
    public void updateChain() { 
        // update your desired fields
        ...
        this.getParent().ifPresent(p -> p.updateChain());
    }
}

然后您可以使用以下 Hibernate 拦截器来强制更新脏实体的所有父级。

public class EntityChainForceUpdateHibernateInterceptor extends EmptyInterceptor {

    @Override
    public void preFlush(Iterator entities) {
        entities.forEachRemaining(e -> {
            if (BaseEntity.class.isAssignableFrom(e.getClass())) {
                BaseEntity b = (BaseEntity) e;
                b.getParent().ifPresent(p -> p.updateChain());
            }
        });
    }
}

要向 Spring 注册此拦截器,只需将以下行添加到 application.properties

spring.jpa.properties.hibernate.ejb.interceptor=com.your.package.EntityChainForceUpdateHibernateInterceptor

这是一个如何覆盖 getParent() 的示例。

@MappedSuperclass
public abstract class BaseEntity {

    @Column(name = "last_updated_at")
    private Instant lastUpdatedAt;

    public Optional<BaseEntity> getParent() {
        return Optional.empty();
    }
    public void updateChain() {
        this.lastUpdatedAt = Instant.now();
        this.getParent().ifPresent(p -> p.updateChain());
    }
}

@Entity
public class ParentEntity extends BaseEntity {
    @Id
    @Column(name = "id")
    private String id;

    @Column(name = "something")
    private String somethingToUpdateWhenChildSaved;

    @Override
    public void updateChain() {
        this.somethingToUpdateWhenChildSaved = "anything";
        super.updateChain();
    }
}

@Entity
public class ChildEntity extends BaseEntity {

    @ManyToOne
    @JoinColumn(name = "parent_id", referencedColumnName = "id")
    private ParentEntity parent;

    @Override
    public Optional<BaseEntity> getParent() {
        return Optional.of(this.parent);
    }
}

关于java - @PreUpdate 更新时不保存父对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45679366/

相关文章:

java - Stream API - 如果 filter() 紧随其后,sorted() 操作如何工作?

java - 在Spring MVC中,bean反序列化和验证涉及的对象之间是什么关系?

java - 正常字段没有类型为 java.lang.String 的合格 bean 异常

java.lang.ClassCastException : java. util.ArrayList 无法转换为 java.lang.String

java - 编译后的kt文件是由字节码组成的?

java - eclipse - 输出

java - 在单个流中组合 allMatch、noneMatch 和 anyMatch

java - Hibernate注解哪里只适用于1对多关系?

java - 使用在不同的基于 java 的 Spring 配置文件中定义的原型(prototype) beans

java - 如何禁用 eclipse 上的控制台日志