【发布时间】:2015-10-13 12:34:16
【问题描述】:
我正在通过添加 CDI 重构旧模块。
我以
结尾public interface ApiFactory {
...
}
public class ApiFactorySp
implements ApiFactory {
@Inject
UrlProducer urlProducer; // <-- Does not get injected
...
}
和
public interface UrlProducer {
public String getUrl();
}
@Alternative
public class UrlProducerTest
implements UrlProducer {
@Override
public String getUrl() {
return "https://myTestEnv.mydomain/myWebApp";
}
}
为了测试,我在META-INF中创建了一个beans.xml文件:
<beans
xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee
http://xmlns.jcp.org/xml/ns/javaee/beans_1_1.xsd"
bean-discovery-mode="all">
<alternatives>
<class>myOrg.myProject.myPackage.UrlProducerTest</class>
</alternatives>
</beans>
为了测试它,我正在做in this blog所示的操作
public class WeldContext {
public static final WeldContext INSTANCE = new WeldContext();
private final Weld weld;
private final WeldContainer container;
private WeldContext() {
this.weld = new Weld();
this.container = weld.initialize();
Runtime.getRuntime().addShutdownHook(new Thread() {
@Override
public void run() {
weld.shutdown();
}
});
}
public <T> T getBean(Class<T> type) {
return container.instance().select(type).get();
}
}
和
public class WeldJUnit4Runner extends BlockJUnit4ClassRunner {
public WeldJUnit4Runner(Class<Object> clazz) throws InitializationError {
super(clazz);
}
@Override
protected Object createTest() {
final Class<?> test = getTestClass().getJavaClass();
return WeldContext.INSTANCE.getBean(test);
}
}
现在,当我尝试测试逻辑时,我会这样做
@RunWith(WeldJUnit4Runner.class)
public class MyTest {
@Inject
UrlProducer urlProducer;
@Inject
ApiFactory apiFactory;
@Test
public void test() {
apiFactory.doSomethingThatRequiresUrlProducer();
}
}
当我运行它时,两个测试属性都被注入,但我得到了 NPE,因为 apiFactory 实例内的 urlProducer 属性尚未分配值。
为什么 Weld 无法识别 ApiFactory 中的 @Inject 属性?
JDK 7、Weld 2.2.10、Junit 4.12
更新:发布问题后,开始尝试一个更简单的全新项目(只有两个接口和三个类)。使用 Weld “独立”并没有解决问题,使用 CDI-Unit 确实解决了它。
然后我修改了我的原始项目以使用 CDI-Unit,但它没有任何改进。之后,我将ApiFactory 中的UrlProducerTest 注入从字段更改为构造函数(即定义@Inject ApiFactory(UrlProducer urlProducer) 构造函数)解决了它。我仍然没有尝试使用“独立”焊接(即明天)的这种解决方案,但我仍然想知道为什么现场注入不起作用。
【问题讨论】:
-
你确定NPE不是因为没有注入apiFactory吗?
-
@A.Panzer 是的,我确定
apiFactory被注入了。但是当我要回复时,我认为我很确定,因为使用UrlProducer的代码是 在构造函数中 :-*> 并且它会导致在实例化过程中引发异常......我明天必须检查它,但也许你想重写你的答案以考虑到这一点。