使用 Spring JPA / Hibernate 进行条件插入

Posted

技术标签:

【中文标题】使用 Spring JPA / Hibernate 进行条件插入【英文标题】:Conditional insert with Spring JPA / Hibernate 【发布时间】:2016-05-06 08:52:28 【问题描述】:

我正在开发一个在集群环境中运行的项目,其中有许多节点和一个数据库。该项目使用 Spring-data-JPA (1.9.0) 和 Hibernate (5.0.1)。我无法解决如何防止重复行问题。

为了举例,这里有一个简单的表格

@Entity
@Table(name = "scheduled_updates")
public class ScheduledUpdateData 
    public enum UpdateType 
        TYPE_A,
        TYPE_B
    

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Column(name = "id")
    private UUID id;

    @Column(name = "type", nullable = false)
    @Enumerated(EnumType.STRING)
    private UpdateType type;

    @Column(name = "source", nullable = false)
    private UUID source;

重要的是有一个UNIQUE(type, source) 约束。

当然还有匹配的示例存储库:

@Repository
public class ScheduledUpdateRepository implements JpaRepository<ScheduledUpdateData, UUID> 
    ScheduledUpdateData findOneByTypeAndSource(final UpdateType type, final UUID source);

    //...

此示例的想法是,系统的某些部分可以插入行以安排定期运行的某些内容,在所述运行之间可多次运行。无论什么东西实际运行,它都不必担心在同一东西上运行两次。

如何编写一个有条件地插入该表的服务方法?我尝试过的一些不起作用的方法是:

    Find > Act - 服务方法将使用存储库来查看条目是否已存在,然后根据需要更新找到的条目或保存新条目。这不起作用。 尝试插入 > 如果失败则更新 - 服务方法将尝试插入,捕获由于唯一约束引起的异常,然后进行更新。这不起作用,因为事务已经处于回滚状态,并且无法在其中进行进一步的操作。

    带有“INSERT INTO ... WHERE NOT EXISTS ...”的本机查询* - 存储库有一个新的本机查询:

    @Repository
    public class ScheduledUpdateRepository implements JpaRepository<ScheduledUpdateData, UUID> 
        // ...
    
        @Modifying
        @Query(nativeQuery = true, value = "INSERT INTO scheduled_updates (type, source)" +
                                           " SELECT :type, :src" +
                                           " WHERE NOT EXISTS (SELECT * FROM scheduled_updates WHERE type = :type AND source = :src)")
        void insertUniquely(@Param("type") final String type, @Param("src") final UUID source);
    
    

    不幸的是,这也不起作用,因为 Hibernate 似乎首先执行WHERE 子句使用的SELECT - 这意味着最终会尝试多次插入,从而导致违反唯一约束。

我绝对不知道 JTA、JPA 或 Hibernate 的许多细节。关于如何跨多个 JVM 插入具有唯一约束(不仅仅是主键)的表中的任何建议?

编辑 2016-02-02

使用 Postgres (2.3) 作为数据库,尝试使用 隔离级别 SERIALIZABLE - 遗憾的是,这仍然会导致违反约束的异常。

【问题讨论】:

尝试将@version注解添加到实体和版本字段到表。 重试整个事务(第二次更新而不是插入)。 @sky_light 不幸的是,@Version 没有帮助,因为当两个事务认为该行尚不存在时会发生错误。 @DraganBozanovic 在整个项目中,单个事务可能会更新多个表,其中任何一个或所有表都可能具有独特的约束,因此重试整个事务听起来需要引入一个全新的框架层在现有的 @Services 之上 - 这似乎不正确 如果serviceA在一个事务中运行,并调用serviceB在自己的事务中运行(即使用REQUIRES_NEW),那么您可以在serviceA中捕获serviceB抛出的异常,并且不会回滚serviceA。 【参考方案1】:

Hibernate 及其查询语言支持插入语句。因此,您实际上可以使用 HQL 编写该查询。浏览此处获取更多信息。 http://docs.jboss.org/hibernate/orm/5.0/userguide/html_single/Hibernate_User_Guide.html#_hql_syntax_for_insert

【讨论】:

虽然这在理论上可以回答这个问题,it would be preferable 在此处包含答案的基本部分,并提供链接以供参考。【参考方案2】:

这听起来像是一个 upsert 案例,可以按照 here 的建议处理。

【讨论】:

【参考方案3】:

Find > Act - 服务方法将使用存储库查看条目是否已存在,然后根据需要更新找到的条目或保存新条目。这不起作用。

为什么这不起作用?

你考虑过“乐观锁”吗?

这两个帖子可能会有所帮助:

https://www.baeldung.com/jpa-optimistic-locking

https://www.baeldung.com/java-jpa-transaction-locks

【讨论】:

这些都不能解决 update-if-exists-insert-otherwise 问题。锁定仅适用于单个记录。如果在初始查询时该记录尚不存在,则无需锁定任何内容。锁定整个表是可能的,但这会造成严重的性能瓶颈。更好的方法是在数据库中创建一个单记录“锁定”表,并将其用作信号量。详情见我的回复。【参考方案4】:

您正在尝试确保一次只有 1 个节点可以执行此操作。 最好的(或至少与数据库无关的)方法是使用“锁定”表。 该表将只有一行,并将充当信号量以确保串行访问。

确保此方法包含在事务中

// this line will block if any other thread already has a lock
// until that thread's transaction commits
Lock lock = entityManager.find(Lock.class, Lock.ID, LockModeType.PESSIMISTIC_WRITE);

// just some change to the row, it doesn't matter what
lock.setDateUpdated(new Timestamp(System.currentTimeMillis()));  
entityManager.merge(lock);
entityManager.flush();

// find your entity by unique constraint
// if it exists, update it
// if it doesn't, insert it

【讨论】:

以上是关于使用 Spring JPA / Hibernate 进行条件插入的主要内容,如果未能解决你的问题,请参考以下文章

spring.jpa.hibernate.hbm2ddl 和 spring.jpa.hibernate.ddl 之间的区别

使用 Spring JPA / Hibernate 进行条件插入

Spring Hibernate JPA 联表查询 复杂查询

使用 JPA 和 Spring 加载选定的 Hibernate 实体

Spring、Spring Security、JPA、MySQL、Hibernate 配置

Spring+Hibernate+JPA的多数据库