我现在正在使用Guice进行JSF Web应用程序的JUnit测试。假设我有几个类:
class ServiceA
{
@Inject
ServiceB serviceB;
//do something using serviceB
...
}
interface ServiceB
{
//APIs in ServiceB
...
}
class ServiceBImpl implements ServiceB
{
@Inject
ServiceC serviceC;
//do something using serviceC
...
}
interface ServiceC
{
//APIs in ServiceC
...
}
class ServiceCImpl implements ServiceC
{
//do something
...
}
现在我使用Guice为 ServiceA 构建模块。我的模块看起来像下面的
class ServiceAModule implements Module
{
@Override
public void configure(Binder binder) {
binder.bind(ServiceB.class).to(ServiceBImpl.class);
}
}
在我的JUnit测试类中,我使用Guice创建了注入,它看起来像:
public ServiceATest
{
ServiceA serviceA;
@Before
public void before()
{
Injector injector = Guice.createInjector(new ServiceAModule());
serviceA = injector.getInstance(ServiceA.class);
}
//do the test
...
}
现在的问题是ServiceATest对“ServiceC”一无所知,也不会注入ServiceB。
当然我可以构建另一个模块来在ServiceB中引入ServiceC的注入。但这肯定会破坏代码分层。如果ServiceC是由另一个人开发的,我对ServiceC一无所知怎么办?这应该不是一个好的解决方案。
Guice可以处理这种多层注射吗?
答案 0 :(得分:2)
除非您正在进行集成测试,否则不应进行多层注入。
在单元测试ServiceATest中,您应该注入ServiceB mock并且不关心ServiceC和guice模块。你可以使用mockito的@InjectMocks而不是Guice。
如果您正在进行集成测试,那么您应该确定特定测试中涉及哪些组件。你应该模拟测试范围之外的所有组件,可能你会在测试用例中使用额外的ad-hock MockModule来绑定模拟的依赖项。
顺便说一下,你可以在单元测试中注入bean,而不是使用getInstance,即:
public ServiceATest
{
@Inject
ServiceA serviceA;
@Before
public void before()
{
Injector injector = Guice.createInjector(new ServiceAModule());
injector.injectMembers(this);
}
}