当使用违反唯一约束的实体调用 persist() 时没有持久性异常

Posted

技术标签:

【中文标题】当使用违反唯一约束的实体调用 persist() 时没有持久性异常【英文标题】:No Persistence Exception when persist() is called with an entity that violates a unique constraint 【发布时间】:2012-08-22 17:35:01 【问题描述】:

我对我的实体的一个属性有唯一的约束:

Table(uniqueConstraints = @UniqueConstraint(name = "UniqueByEmail", columnNames = "email"))
public class Player implements Serializable 
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private long id;

    @Column(name = "email")
    private String email;
    ...

在我的 DAO 中,当某些东西试图创建一个违反此约束的新玩家时,我想抛出一个适当的异常:

public Player create(String email, String password) 
    Player player = new Player(email, password);
    try 
        em.persist(player);
     catch (Exception e) 
        System.err.println("I GOT AN EXCEPTION : " + e.getClass());
    
    return player;

不幸的是,什么也没发生。当我在实体管理器上调用 flush() 时,由于此调用,我得到了持久性异常,但当我调用 persist() 时却没有。

这种行为与休眠的行为不同,所以我认为我配置错误。

非常感谢任何帮助。

持久性.xml: org.eclipse.persistence.jpa.PersistenceProvider 模型.GameInstanceAccount 播放器

spring applicationContext.xml:

<tx:annotation-driven transaction-manager="transactionManager"/>
<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
    <property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>

<bean id="entityManager" class="org.springframework.orm.jpa.support.SharedEntityManagerBean">
    <property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>

<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
    <property name="persistenceUnitName" value="persistence-unit" />
    <property name="jpaDialect" ref="jpaDialect" />
    <property name="jpaVendorAdapter">
        <bean class="org.springframework.orm.jpa.vendor.EclipseLinkJpaVendorAdapter" />
    </property>
</bean>

<bean id="jpaDialect" class="org.springframework.orm.jpa.vendor.EclipseLinkJpaDialect" />

<bean id="jpaVendorAdapter" class="org.springframework.orm.jpa.vendor.EclipseLinkJpaVendorAdapter">
    <property name="generateDdl" value="true" />
    <property name="databasePlatform" value="org.eclipse.persistence.platform.database.HSQLPlatform" />
</bean>

【问题讨论】:

【参考方案1】:

当我在实体管理器上调用 flush() 时,我得到了持久性 此调用导致异常,但当我调用 persist() 时不会发生异常

这是预期的行为。持久调用仅将瞬态实体附加到当前持久会话,使其受到管理。该约束适用于数据库,因此只有在刷新持久性会话并将新实体插入数据库时​​才会应用。

这个blog总结如下:

一个新创建的实体类实例被传递给persist 方法。此方法返回后,对实体进行管理和计划 用于插入数据库。它可能发生在或之前 事务提交或调用 flush 方法时。

【讨论】:

这可能只是休眠状态,例如在调用 persist 时触发插入查询,并在出现问题时抛出持久异常。我想要并且需要这种行为。是否可以从 eclipselink 获取? “hibernate 例如在调用persist 时触发插入查询” - 这不一定是真的。这取决于用于生成标识符的策略。

以上是关于当使用违反唯一约束的实体调用 persist() 时没有持久性异常的主要内容,如果未能解决你的问题,请参考以下文章

尝试插入具有 1:N 关系的实体时,重复键值违反 EntityFramework 中的唯一约束“PK_Users”错误

在持久化实体之前检查是不是违反约束

如何使用hibernate避免遇到违反唯一约束的问题?

Azure Cosmos DB:使用 UpsertDocumentAsync 违反唯一索引约束

Hibernate JPA 关联关系

IntegrityError:重复键值违反唯一约束