【问题标题】:Vaadin JPAContainer - no transaction is in progressVaadin JPAContainer - 没有正在进行的事务
【发布时间】:2015-04-22 06:48:22
【问题描述】:

我一直在尝试将 Vaadin JPAContainer 与 JBoss 中的 JTA 数据源一起使用,但在提交 FieldGroup 时我不断收到此错误:“没有正在进行的事务”

这是我的 persistence.xml

    <persistence xmlns="http://java.sun.com/xml/ns/persistence"
             xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"
             version="2.0">
   <persistence-unit name="Pagamento">
      <provider>org.hibernate.ejb.HibernatePersistence</provider>
      <jta-data-source>java:jboss/datasources/Pagamento</jta-data-source>  

      <class>br.com.edu.entidades.Usuario</class>

      <properties>
         <property name="hibernate.dialect" value="org.hibernate.dialect.PostgreSQLDialect"></property>         
         <property name="hibernate.cache.use_second_level_cache" value="false"></property>
         <property name="hibernate.hbm2ddl.auto" value="update"></property>
         <property name="hibernate.show_sql" value="false"></property>  
         <property name="hibernate.format_sql" value="true"></property>
      </properties>

   </persistence-unit>
</persistence>

我还在使用自定义 EntityProvider 来支持 JTA:

public class EduEntityProvider extends MutableLocalEntityProvider<Usuario> {     

    private EntityManager em = HibernateUtils.entityManager;

    public EduEntityProvider() {
        super(Usuario.class);
        setEntityManager(em);

        setEntitiesDetached(false);
        setTransactionsHandledByProvider(false);
    }

    @Override
    @TransactionAttribute(TransactionAttributeType.REQUIRED)
    protected void runInTransaction(Runnable operation) {
        super.runInTransaction(operation);
    }    

}

实体:

@SuppressWarnings("serial")
@Entity
@Table(name="usuario")
public class Usuario extends EduPersistentEntity {

    @Id
    @Column(name="id")
    @VisivelTabela(false)
    private String id;

    @Column(name="nome")
    private String nome;

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

    @Column(name="senha")
    private String senha;

    @Column(name="matricula", unique=true)
    private String matricula;

    gets... 
    sets...
}

如果我将提供者处理的事务设置为 true

"setTransactionsHandledByProvider(false);", i'll get the error 
" A JTA EntityManager cannot use getTransaction()".

如果有人能给我指出一个可行的例子,或者只是给我一个关于我做错了什么的提示,将不胜感激。

【问题讨论】:

  • super.runInTransaction(operation) method 长什么样子?
  • 您有使用 JPAContainer 的特定理由吗?对于任何严肃的使用,我建议使用@Stateless EJB 来隐藏 UI 层的 JPA 内容。然后只需使用 BeanItemContainer 将您的实体对象列表传递给 UI 组件。或者,如果您非常关心应用程序的性能,请查看 Viritin 插件及其lazy loading features。那些性能比 JPAContainer 更好(CPU 和内存)并且更易于使用。
  • @wypieprz, super.runInTransaction(operation) 产生相同的结果...但是谢谢
  • @mstahv,我确实考虑过,但这会破坏我的整个应用程序,而是手动控制 EntityManager 并传递给 JPAContainer,我知道这不是最好的做法,但是它解决了问题,实际上并不需要大量重构。
  • 不使用 JPAContainer 或使用 Java EE 应用服务器?应该可以逐步从 JPAContainer 转向 Facade+BeanItemContainer/Viritin 风格的架构。 EJB 只会让数据库访问比普通的 EntityManager 更容易。

标签: jpa jboss vaadin jta jpacontainer


【解决方案1】:

以下错误不言自明:

JTA EntityManager 不能使用 getTransaction()

换句话说,当 JTA 由 persistence.xml 中的数据源指定时,您不能调用 EntityTransaction.getTransaction()


以下是事务管理的一般规则,因此您可以遵循其中之一并根据需要采用提供程序。

  • 资源本地事务(Java EE / Java SE)

    • 应用程序管理的实体管理器

       @PersistenceUnit(unitName="...")
       EntityManagerFactory emf;
      
       public void foo() {
           em.getTransaction.begin(); //explicit begin
           ...
           em.getTransaction.commit(); // explicit commit
       }
      
  • JTA 事务(仅限 Java EE)

    • 应用程序管理的实体管理器(bean 管理的事务)

       @PersistenceUnit(unitName="...")
       EntityManagerFactory emf;
       EntityManager em = emf.createEntityManager();
      
       @Resource
       UserTransaction tx;
      
       public void foo() {
           tx.begin(); //explicit begin
           ...
           tx.commit(); // explicit commit
       }
      
    • 通过 EJB(容器管理事务)的容器管理实体管理器

        @PersistenceContext(unitName="...")
        EntityManager em;
      
        @TransactionAttribute(TransactionAttributeType.REQUIRED) //default
        public void foo() {
            // implicit begin
            ...
            // implicit commit
        }
      
    • 通过 CDI 的容器管理实体管理器(容器管理事务)

        @PersistenceContext(unitName="...")
        EntityManager em;
      
        @Transactional(TxType.REQUIRED) //default
        public void foo() {
            // implicit begin
            ...
            // implicit commit
        }
      

【讨论】:

  • 我明白了,但问题在于 Vaadin 的 JPAContainer。 vaadin FieldGroup 正在尝试持久化实体,这不是我的代码。
  • 我简直不敢相信。您的示例看起来与 Matti Tavhonen 的 Using JPA Container 2.0 in a JEE6 server 博客文章几乎相同。您是否尝试过注入 EntityManager 而不是 HibernateUtil 的分配并按照 reference example 中指定的方式初始化类?
猜你喜欢
  • 2012-11-20
  • 1970-01-01
  • 1970-01-01
  • 2014-04-25
  • 2012-07-29
  • 1970-01-01
  • 2013-02-22
  • 1970-01-01
  • 2019-06-28
相关资源
最近更新 更多