java - 有没有办法在遇到异常时强制事务回滚?

标签 java hibernate ejb

当程序遇到异常时,RollBackOnly 变为 True。

即使遇到异常,我如何“设置”此回滚为 False。

@Resource
protected SessionContext context;

Public void creerNewEntity(final Entity myEntity) {
try {
    this.em.persist(myEntity);
    this.em.flush();

    } catch (final EntityExistsException e) {
      System.out.println((this.context.getRollbackOnly())); // It s has a true value
      throw new MyException("TODO BLABLA", e);
    }
 }  

当程序抛出此异常“MyException”时,我通过设置新的 Id 来更改对象 myEntity,然后再次调用 creerNewEntity().

不幸的是,它不起作用,我得到了这个异常“javax.persistence.PersistenceException:org.hibernate.HibernateException:代理句柄不再有效”,我认为因为RollBack有一个true 值,如何更改回滚才能使其正常工作?

谢谢。

最佳答案

可能没有一种简单的方法可以做到这一点,因为 EJB 设计的重点是您不关心这些事情。事务内的第一个错误使其无效 -> 回滚。这就是规则。

如果您想要一些特殊的东西,那么可以从 session 中获取数据库连接,并使用纯 SQL 而不是 EJB 来修改数据。这样,您就可以尝试INSERT一个新实例并自己处理所有异常。当插入成功后,您可以使用EJB加载新创建的对象,将其添加到 session 中。

也就是说,我不确定您想通过上面的代码实现什么目的。当您无法在数据库中创建新实例时,忽略它就像“我不关心我的产品质量”。也许您尝试解决错误只是应用程序设计不良的症状。退后一步,考虑一下你在做什么以及为什么。也许如果您告诉我们更多关于您想要忽略所有错误(甚至是真正致命的错误)的原因,我们就能指出更好的解决方案。

编辑因此您会得到javax.persistence.EntityExistsException,这意味着您尝试保存同一实体两次。这可能意味着很多事情:

  1. 您在不同的 session 中加载了该 Bean,现在尝试将其保存在第二个 session 中。由于新 session 无法知道该 bean 是否存在,因此它会尝试再次创建它。
  2. 您没有像您应该的那样从 session 中加载 bean,而是作弊并手动创建了一个新实例。当然, session 管理器现在认为这是一个新 bean。

正确的解决方案取决于您需要实现的目标。如果您修改了 myEntity 并需要保存更改,请使用 em.merge()。然后,EM 将检查该对象是否已存在,如果存在,它将执行 SQL UPDATE 而不是 INSERT

如果您只想为其他类提供一个有效实体,那么您需要从数据库中获取它。如果数据库返回null,则需要创建一个新实例并持久化,然后返回。

另请参阅:JPA EntityManager: Why use persist() over merge()?

关于java - 有没有办法在遇到异常时强制事务回滚?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26102635/

相关文章:

java - 如何在Java中分别绑定(bind)IPv6和IPv4

java - @transactional 方法不会在我们进入方法后立即启动 session

Hibernate Criteria API - 访问连接属性

java - WAS Liberty 17.0.0.3 上的非持久性计时器创建

json - 使用 Jackson 将 JPA 实体序列化为 JSON

java - JRebel 与最新 weblogic 的兼容性

java - 在 Java 中解释/开发/输入/鼠标

java - 发布 spring boot 的项目列表

java - 同步spring事务

java - 使用 "crontab syntax"安排 EJB 任务