JPA 事务立即将插入存储在数据库中

JPA transaction stores insertions inmediatly in DataBase

当我在交易 (TransactionAttributeType.REQUIRED) 方法中保存一个新实体时 createEntities 实体在交易完成之前立即存储在数据库中。我预计事务已完成并且更改会在 createEntities 方法完成时传播到数据库,但如果我在 syso 行中调试并暂停执行,我可以在外部应用程序(例如 Toad)中看到数据库中的更改。

是我猜错了还是配置方法有误?

我的代码如下:

@Stateless
public class MyClass {
    @Inject
    private MyService myService;
    
    @TransactionAttribute(TransactionAttributeType.NOT_SUPPORTED)
    public void aMethod() {
        // don't want a transaction because this method does a lot of calculations
        // and lasts a lot of time resulting in transaction timeout otherway
        
        createEntities();
    }

    @TransactionAttribute(TransactionAttributeType.REQUIRED)
    public void createEntities() {
        myService.createEntity(new MyEntity(1, "111"));
        System.out.println("Paused execution, checked DB and saw the entity was already inserted");
        myService.createEntity(new MyEntity(2, "222"));
        System.out.println("Paused execution, checked DB and saw the entity was already inserted");
    }
}

@Stateless
public class MyService {
    @PersistenceContext
    private EntityManager em;
    
    public void createEntity(MyEntity entity) {
        em.merge(entity);
    }
}

仅当从另一个 Bean 调用方法时,@TransactionAttribute 才有效。
如果您从其他 Bean 调用 aMethod(),该调用将被拦截以暂停最终活动的事务。
然后 createEntities() 在没有事务活动的情况下被调用,但是这个调用不会被拦截,因为它是从 MyClass-Bean 内部调用的。
因此不会启动任何事务。
createEntity()-方法未注释,因此 TransactionAttribute.Required 处于活动状态。
因为这个方法不是从同一个 Bean 调用的,并且没有事务处于活动状态,所以容器将启动一个新事务。此事务在方法结束时提交。

有关容器管理事务的更多信息:https://docs.oracle.com/javaee/6/tutorial/doc/bncij.html