在每次单元测试后创建一个bean的新实例

geo*_*ydv 8 java spring unit-testing

我是spring框架的新手,并且使用Spring Context对其依赖注入功能有疑问.

这是我正在尝试编写集成测试的类:

public class UserService {

private Validator validator;
private UserRepository userRepository;
private Encryptor encryptor;
private MailService mailService;

...

public void registerUser(User user) {
    user.setPassword(encryptor.encrypt(user.getPassword()));

    Errors errors = new BindException(user, "user");
    validator.validate(user, errors);

    if (errors.getErrorCount() == 0) {
        userRepository.addUser(user);
        mailService.sendMail(user.getEmail());
    }
}
Run Code Online (Sandbox Code Playgroud)

在我的测试中(使用Mockito)我想确保调用这四个项目,所以我创建了如下测试:

public void testRegisterCallsValidateInValidator() {
    userService.registerUser(testUser);
    verify(userService.getValidator(), times(1)).validate(any(User.class), any(Errors.class));
}
Run Code Online (Sandbox Code Playgroud)

然而,所有测试都失败了,说我多次调用该方法.我唯一的猜测是UserService bean在所有测试开始时创建一次,但在每次测试后都不会重新加载.

在我的测试配置中,我使用以下xml来决定要注入的bean:

<bean id="userService" class="be.kdg.coportio.services.UserService">
    <property name="validator" ref="validator"/>
    <property name="userRepository" ref="userRepository"/>
    <property name="encryptor" ref="encryptor"/>
    <property name="mailService" ref="mailService"/>
</bean>
Run Code Online (Sandbox Code Playgroud)

有任何想法吗?

gpe*_*che 30

您正在重复使用上下文,为了使测试彼此独立,您可能需要在每次测试后刷新上下文以重置所有内容.

我假设您使用的是Junit 4.5+.它与其他测试框架类似.

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"mycontext.xml"})
@DirtiesContext(classMode = ClassMode.AFTER_EACH_TEST_METHOD)
public class MyTestClass {
...
    // my tests    
...
}
Run Code Online (Sandbox Code Playgroud)

@DirtiesContext如果需要"修复"的方法很少,你可以把方法放在方法级别,但如果你使用Spring,你最好的选择是在每次测试后都这样做.

无论如何,我不认为你应该在集成测试中使用模拟/间谍:

  • 在单元测试中,使用模拟(如果需要)并手动注入.在这里,您要验证测试bean作为一个单元的行为,因此您可以使用模拟将其与其余bean隔离.这也有一个优点,即JUnit通过为每个测试使用不同的测试类实例来隔离您的测试,因此除非您使用static或其他测试不友好的实践,否则一切都会正常工作.

  • 在集成测试中,使用真正的bean并让Spring注入.这里的目标是验证bean彼此之间/与环境(数据库,网络......)的良好交互.您希望在此处隔离bean,因此不应使用模拟.

有关测试的详细说明,请参阅Spring文档.


Eug*_*gen 9

要明确区分单元和集成测试(跳过关于每个类别的含义的争论) - 您可以通过两种方式测试您的服务:

  • 通过集成测试 - 您启动整个Spring Context并将服务作为单例bean进行测试.
  • 通过单元测试 - 您只需自己初始化服务,模拟需要模拟的内容,而不需要Spring.

我的建议是不要混合Spring和模拟如果你可以帮助它 - 保持Mockito进行单元测试(这是你需要的东西)并使用集成测试来引导整个Spring上下文来测试其他东西 - 持久性问题,交易等

你不需要Spring来模拟一个类的协作者,并与Mockito进行简单的交互测试.

  • "我的建议是,如果你能帮助它,就不要混合使用Spring和mocks".如果您有setter或使用Spring的`ReflectionTestUtils.setField()`方法,请手动注入您的模拟.如果您的服务在每次测试中都发生变化,则无需启动弹簧上下文 (2认同)