【发布时间】:2017-02-27 12:55:32
【问题描述】:
在 java-spring web-app 中,我希望能够动态注入 bean。 例如,我有一个具有 2 个不同实现的接口:
在我的应用中,我使用一些属性文件来配置注入:
#Determines the interface type the app uses. Possible values: implA, implB
myinterface.type=implA
我的注入实际上是根据属性文件中的属性值有条件地加载的。例如,在这种情况下,myinterface.type=implA 无论我在哪里注入 MyInterface,将被注入的实现都是 ImplA(我通过扩展 Conditional annotation 实现了这一点)。
我希望在运行时 - 一旦更改属性,将发生以下情况(无需重新启动服务器):
- 将注入正确的实现。例如,当设置
myinterface.type=implB时,ImplB 将被注入到任何使用 MyInterface 的地方 - Spring Environment 应该用新值刷新并重新注入到 bean 中。
我想刷新我的上下文,但这会产生问题。 我想也许可以使用设置器进行注入,并在重新配置属性后重新使用这些设置器。有这种要求的工作实践吗?
有什么想法吗?
更新
正如一些人建议的那样,我可以使用一个工厂/注册表,它包含两个实现(ImplA 和 ImplB)并通过查询相关属性返回正确的一个。 如果我这样做了,我还有第二个挑战——环境。例如,如果我的注册表如下所示:
@Service
public class MyRegistry {
private String configurationValue;
private final MyInterface implA;
private final MyInterface implB;
@Inject
public MyRegistry(Environmant env, MyInterface implA, MyInterface ImplB) {
this.implA = implA;
this.implB = implB;
this.configurationValue = env.getProperty("myinterface.type");
}
public MyInterface getMyInterface() {
switch(configurationValue) {
case "implA":
return implA;
case "implB":
return implB;
}
}
}
一旦属性发生变化,我应该重新注入我的环境。有什么建议吗?
我知道我可以在方法中查询该 env 而不是构造函数,但这会降低性能,而且我想考虑一个用于重新注入环境的 ider(再次,也许使用 setter 注入?)。
【问题讨论】:
-
如何将属性中的值更改为 myinterface.type=implB,以及系统如何知道您完成/完成更改了值
-
此外,应用程序应该如何判断已经注入的依赖项在使用中并且您强制交换实现的情况?这种方法的主要目的是什么?
-
@kuhajeyan 假设我知道(例如使用 WatchService),这不是这里的挑战。
-
@forhas,请查看此链接:stackoverflow.com/questions/12800769/…。它可能对你有帮助。但它真正黑暗的方式是在运行时交换 bean 实现。您的问题应该通过正确的设计模式来解决。
-
@forhas,类似问题有很好的答案:stackoverflow.com/questions/21221125/…
标签: java spring polymorphism spring-environment