我正在使用Wildfly 8.2容器中的Java EE 7应用程序,该容器包含两个数据源中存在的一些entite。例如:
我有一个Setting
实体的jar:
@Entity
public class Setting {
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private Long id;
@NotNull
private String name;
private String value;
getters/setters...
}
和一个bean,它有几种方法可以通过条件查询从数据库中检索和保存这个实体:
@Stateless
public class SettingRepository {
@Inject
private Logger logger;
@Inject
private EntityManager entityManager;
public Setting findByName(@NotNull String name) {
logger.trace("Getting setting by name: name=" + name);
CriteriaBuilder cb = entityManager.getCriteriaBuilder();
CriteriaQuery<Setting> cq = cb.createQuery(Setting.class);
Root<Setting> table = cq.from(Setting.class);
cq.where(cb.equal(table.get(Setting_.name), name));
TypedQuery<Setting> query = entityManager.createQuery(cq);
List<Setting> results = query.getResultList();
Setting setting = null;
if (results.size() > 0)
setting = results.get(0);
logger.trace("Got setting: " + setting);
return setting;
}
...
}
我不想通过@Producer在一个应用程序中提供EntityManager
和Logger
实例,该应用程序在类路径中包含此jar:
@Produces @DataSource1 @PersistenceContext(unitName = "pu1")
private EntityManager entityManager1;
@Produces @DataSource2 @PersistenceContext(unitName = "pu2")
private EntityManager entityManager2;
@Produces
private Logger produceLogger(InjectionPoint injectionPoint) {
return LogManager.getLogger(injectionPoint.getMember().getDeclaringClass().getName());
}
有没有办法在注入点配置SettingRepository并告诉它使用特定的实体管理器(@DataSource1
或@DataSource2
)?
与此类似:
@Inject @DataSource1
private SettingRepository settingRepository;
答案 0 :(得分:1)
我选择的路线是这个:
我创建了一个限定符
@Qualifier
@Retention(RUNTIME)
@Target({ METHOD, FIELD, PARAMETER })
public @interface DataSource {
@Nonbinding DataSourceName value() default DataSourceName.D1;
}
注意@Nonbinding
注释告诉容器我不必在注释生成器方法时指定此参数(特别是这个,正则我可以实现泛型生成器,稍后会更多)或注入点。
DataSourceName
枚举,它只列出了所有数据源:
public enum DataSourceName {
D1, D2
}
我还更改了SettingRepository
实现并添加了一个公共初始化方法。
@Dependent
public class SettingRepository {
@Inject
private Logger logger;
private EntityManager entityManager;
public void initialize(EntityManager entityManager) {
this.entityManager = entityManager;
}
...
}
请注意,EntityManager
不再按容器注入。
当我不想注射SettingRepository
时,我只是用这个限定符装饰它,如下:
@Inject @DataSource(DataSourceName.D1)
private SettingRepository settingRepository;
我现在需要的是为SettingRepository
定义一个制作人:
@Stateless
public class TestResourcesForSettings {
@PersistenceContext(unitName = "pu1")
private EntityManager entityManager1;
@PersistenceContext(unitName = "pu2")
private EntityManager entityManager2;
@Inject
private SettingRepository settingRepository;
@Produces @DataSource
public SettingRepository produceSettingRepository(InjectionPoint ip) {
DataSource annotation = ip.getAnnotated().getAnnotation(DataSource.class);
if (annotation.value() == DataSourceName.D1)
settingRepository.initialize(entityManager1);
else if (annotation.value() == DataSourceName.D2)
settingRepository.initialize(entityManager2);
return settingRepository;
}
瞧,我可以使用两个数据源和一个存储库实现(注意:数据源必须是XA)。 如果有人发现此选择有任何问题,请告诉我。
答案 1 :(得分:0)
是的!
事实上,就生产者方法和限定符而言,你已经完成了所有艰苦的工作:-)你需要做的就是在注入的EntityManager实例上使用限定符(@DataSource1等)
@Inject @DataSource1
private EntityManager entityManager;
这将确保附加到持久性单元'pu1'的JPA实体管理器被容器注入。