如何动态修改@PersistenceContext 中的unitName

how to dynamically modifying unitName in @PersistenceContext

我正在使用 JPA 2.1,我有这样的东西

public class EntityManagerProducer {
    @Produces
    @PersistenceContext(unitName="first_PU")
    private EntityManager em;
    ...

如何动态修改 @PersistenceContext(unitName = "somer_PU") 中的 unitName 以使用其他实体管理器?这可能吗?

VERY IMPORTANT UPDATE

我有一个已部署的应用程序,persistence.xml 看起来像这样:

<persistence-unit name="db1" transaction-type="JTA">
<jta-data-source>java:/jboss/datasources/PostgresDS</jta-data-source>
<properties>
  <property name="hibernate.hbm2ddl.auto" value="update"/>
  <property name="hibernate.show_sql" value="true"/>
  <property name="hibernate.format_sql" value="true"/>
  <property name="hibernate.dialect" value="org.hibernate.dialect.PostgreSQLDialect"/>
  <property name="hibernate.cache.use_second_level_cache" value="false"/>
  <property name="hibernate.jdbc.batch_size" value="50"/>
  <property name="hibernate.jdbc.batch_versioned_data" value="true"/>
  <property name="hibernate.order_inserts" value="true"/>
  <property name="hibernate.order_updates" value="true"/>
  <property name="hibernate.generate_statistics" value="true"/>
  <property name="javax.persistence.schema-generation.database.action" value="drop-and-create"/>
  <property name="jboss.entity.manager.jndi.name" value="java:app/entitymanager/db1"/>
  <property name="jboss.entity.manager.factory.jndi.name" value="java:app/entitymanagerfactory/db1"/>
</properties>

在运行时,可以修改 persistence.xml 并添加另一个持久性单元,但我需要一种方法来获取我需要的实体管理器,方法是传递持久性单元的名称或其他名称,然后将其用于什么我想。然后我可以提供持久性单元的另一个名称并获得一个不同的实体管理器。当然,我希望交易过程仍然是容器管理的

这是不可能的。如果您需要动态持久性单元名称,则需要完全避免 EE 集成并直接使用 javax.persistence.Persistence class,但我不建议这样做。作为替代方案,您可以为所有持久性单元注入 EntityManager/Factory,然后使用 switch 语句或类似语句 select 正确的。

相信我,您不想修改现有 PersistenceContext 中的 unitName。我几乎可以肯定,您想要实现的只是让第二个 EntityManager 指向不同的数据库,对吗?

那么一个自然的方法是:

public class SomeClass {

    @PersistenceContext(unitName = "first_PU")
    EntityManager firstEntityManager;

    @PersistenceContext(unitName = "somer_PU")
    EntityManager secondEntityManager;

    // ...
}

个人建议

  1. 另请注意,使用 EJB 时,您不必为 EntityManager 创建 Producer 方法,因为键入 @PersistenceContext(...) 就足够了。

  2. 也非常不鼓励以下内容:

    @Produces
    @PersistenceContext(unitName="first_PU")
    private EntityManager em;
    

    因为默认范围是 @Dependent@RequestScoped@TransactionScoped.

  3. 会更好
  4. 您可以使用 CDI 和注释限定符代替硬编码的单元名称:

    @Inject
    private @FirstDB EntityManager firstEntityManager;
    
    @Inject
    private @SecondDB EntityManager secondEntityManager;
    

    但是你需要:

    @Produces
    @RequestScoped // or other
    @FirstDB
    public EntityManager createEntityManagerA() {
        return firstEmf.createEntityManager();
    }
    
    @Produces
    @RequestScoped  // or other
    @SecondDB
    public EntityManager createEntityManagerB() {
        return secondEmf.createEntityManager();
    }
    

您可以找到很多关于 CDI 和生成多个 EntityManagers 的有用信息 here