spring - 如何使用运行时 "qualifier"变量动态注入(inject)服务?

标签 spring service dependency-injection autowired

在给定运行时值的情况下,我找不到注入(inject)组件/服务的简单方法。

我开始阅读@Spring 的文档:http://docs.spring.io/spring/docs/current/spring-framework-reference/html/beans.html#beans-autowired-annotation-qualifiers 但我找不到如何改变传递给 @Qualifier 注释的值。

假设我有一个具有这种接口(interface)的模型实体:

public interface Case {

    String getCountryCode();
    void setCountryCode(String countryCode);

}

在我的客户端代码中,我会执行以下操作:

@Inject
DoService does;

(...)

Case myCase = new CaseImpl(); // ...or whatever
myCase.setCountryCode("uk");

does.whateverWith(myCase);

...我的服务是:

@Service
public class DoService {

    @Inject
    // FIXME what kind of #$@& symbol can I use here?
    // Seems like SpEL is sadly invalid here :(
    @Qualifier("${caze.countryCode}")
    private CaseService caseService;

    public void whateverWith(Case caze) {
        caseService.modify(caze);
    }

}

我希望 caseService 是 UKCaseService(请参阅下面的相关代码)。

public interface CaseService {

    void modify(Case caze);

}

@Service
@Qualifier("uk")
public class UKCaseService implements CaseService {

}

@Service
@Qualifier("us")
public class USCaseService implements CaseService {

}

那么我如何以最简单/优雅/有效的方式通过使用任何一个/所有 Spring 特性来“修复”所有这些,所以基本上没有 .properties,没有 XML,只有注释。 但是我已经怀疑我的 DoService 有问题,因为 Spring 在注入(inject) caseService 之前需要知道“案例”......但是如何在客户端代码不知道 caseService 的情况下实现这一点?! 我想不通...

我已经在这里阅读了几个关于 SO 的问题,但大多数时候,要么他们的需求和/或配置与我不同,要么发布的答案对我来说不够满意(看起来他们'本质上是变通办法或(旧)使用(旧)Spring特性)。

How does Spring autowire by name when more than one matching bean is found? => 仅指类组件类

Dynamically defining which bean to autowire in Spring (using qualifiers) => 真的很有趣,但最详尽的答案(4 票)是......几乎 3 1/2 岁?! (2013 年 7 月)

Spring 3 - Dynamic Autowiring at runtime based on another object attribute =>这里的问题非常相似,但答案看起来真的是一种解决方法,而不是真正的设计模式(如工厂)?而且我不喜欢将所有代码都实现到 ServiceImpl 中...

Spring @Autowiring, how to use an object factory to choose implementation? => 第二个答案似乎很有趣,但它的作者没有扩展,所以虽然我(有点)知道 Java Config 和东西,但我不太确定他在说什么......

How to inject different services at runtime based on a property with Spring without XML => 有趣的讨论,尤其是。答案,但用户设置了我没有的属性。

另请阅读: http://docs.spring.io/spring/docs/current/spring-framework-reference/html/expressions.html#expressions-bean-references => 我找不到关于在表达式中使用“@”的扩展示例。有人知道吗?

编辑: 发现了其他相关的类似问题,没有人得到正确的答案: How to use @Autowired to dynamically inject implementation like a factory pattern Spring Qualifier and property placeholder Spring: Using @Qualifier with Property Placeholder How to do conditional auto-wiring in Spring? Dynamic injection in Spring SpEL in @Qualifier refer to same bean How to use SpEL to inject result of method call in Spring?

工厂模式可能是一个解决方案? How to use @Autowired to dynamically inject implementation like a factory pattern

最佳答案

您可以使用BeanFactory 动态地从上下文中获取您的bean。 :

@Service
public class Doer {

  @Autowired BeanFactory beans;

  public void doSomething(Case case){
    CaseService service = beans.getBean(case.getCountryCode(), CaseService.class)
    service.doSomething(case);
  }
}

附注。 使用国家代码作为 bean 名称看起来有点奇怪。至少添加一些前缀或更好地考虑其他一些设计模式。

如果您仍然希望每个国家/地区都有 bean,我会建议另一种方法。引入注册服务,按国家代码获取所需服务:

@Service
public class CaseServices {

  private final Map<String, CaseService> servicesByCountryCode = new HashMap<>();

  @Autowired
  public CaseServices(List<CaseService> services){
    for (CaseService service: services){
      register(service.getCountryCode(), service);
    }
  }

  public void register(String countryCode, CaseService service) {
    this.servicesByCountryCode.put(countryCode, service);
  }

  public CaseService getCaseService(String countryCode){
    return this.servicesByCountryCode.get(countryCode);
  }
}

示例用法:

@Service
public class DoService {

  @Autowired CaseServices caseServices;

  public void doSomethingWith(Case case){
    CaseService service = caseServices.getCaseService(case.getCountryCode());
    service.modify(case);
  }
}

在这种情况下,您必须将 String getCountryCode() 方法添加到您的 CaseService 接口(interface)。

public interface CaseService {
    void modify(Case case);
    String getCountryCode();
}

或者,您可以添加方法 CaseService.supports(Case case) 来选择服务。或者,如果您无法扩展接口(interface),您可以从某个初始化程序或 @Configuration 类调用 CaseServices.register(String, CaseService) 方法。

更新:忘了提一下,Spring 已经提供了一个不错的 Plugin抽象以重用样板代码来创建像这样的 PluginRegistry

例子:

public interface CaseService extends Plugin<String>{
    void doSomething(Case case);
}

@Service
@Priority(0)
public class SwissCaseService implements CaseService {

  void doSomething(Case case){
    // Do something with the Swiss case
  }

  boolean supports(String countryCode){
    return countryCode.equals("CH");
  }
}

@Service
@Priority(Ordered.LOWEST_PRECEDENCE)
public class DefaultCaseService implements CaseService {

  void doSomething(Case case){
    // Do something with the case by-default
  }

  boolean supports(String countryCode){
    return true;
  }
}

@Service
public class CaseServices {

  private final PluginRegistry<CaseService<?>, String> registry;

  @Autowired
  public Cases(List<CaseService> services){
    this.registry = OrderAwarePluginRegistry.create(services);
  }

  public CaseService getCaseService(String countryCode){
    return registry.getPluginFor(countryCode);
  }
}

关于spring - 如何使用运行时 "qualifier"变量动态注入(inject)服务?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42909283/

相关文章:

java - 对象池、IoC 和工厂 : When and Where?

java - 多个独立组分注入(inject)

java - 如何使用 Spring Data REST 在 OneToMany 关系中添加对象

用于访问餐厅菜单数据的 API

c# - 在 NLog 中使用依赖注入(inject)

java - 将 JAR 作为服务运行时未显示日志

.net - 我如何使用 OpenCover 连接到服务?

spring - 具有自动配置功能的 Spring Data JPA 应用程序的多个数据库

java - 使用 Spring REST 分页对 MongoDB 中的 ObjectId 进行日期范围查询

java - Apache Commons FileUpload Stream API 在 Spring 5 中不起作用