我在单元测试中有任何问题,我有一些与此类似的问题。如果 blargh
函数使用事务性注释,则模拟注入将在 someService 上被覆盖。如果我删除事务,模拟会留在那里。从观察代码来看,当服务中的函数被注释为Transactinal时,Spring 似乎会延迟加载服务,但当它没有时,它会急切地加载服务。这将覆盖我注入的模拟。
有没有更好的方法可以做到这一点?
@Component
public class SomeTests
{
@Autowired
private SomeService someService;
@Test
@Transactional
public void test(){
FooBar fooBarMock = mock(FooBar.class);
ReflectionTestUtils.setField(someService, "fooBar", fooBarMock);
}
}
@Service
public class someService
{
@Autowired FooBar foobar;
@Transactional // <-- this causes the mocked item to be overridden
public void blargh()
{
fooBar.doStuff();
}
}
也许您可以尝试通过以下方式实现测试:
@Component
@RunWith(MockitoJUnitRunner.class)
public class SomeTests
{
@Mock private FooBar foobar;
@InjectMocks private final SomeService someService = new SomeService();
@Test
@Transactional
public void test(){
when(fooBar.doStuff()).then....;
someService.blargh() .....
}
}
我现在无法尝试,因为没有您的配置和相关代码。但这是测试服务逻辑的常用方法之一。
使用 Spring @Profile 功能 - bean 可以关联到某个组,并且可以通过注释激活或停用该组。
查看此博客文章和文档以获取更详细的说明,这是如何定义生产服务和两组模拟服务的示例:
@Configuration
@Profile("production")
public static class ProductionConfig {
@Bean
public InvoiceService realInvoiceService() {
...
}
...
}
@Configuration
@Profile("testServices")
public static class TestConfiguration {
@Bean
public InvoiceService mockedInvoiceService() {
...
}
...
}
@Configuration
@Profile("otherTestServices")
public static class OtherTestConfiguration {
@Bean
public InvoiceService otherMockedInvoiceService() {
...
}
...
}
这就是在测试中使用它们的方法:
@ActiveProfiles("testServices")
public class MyTest extends SpringContextTestCase {
@Autowired
private MyService mockedService;
// ...
}
@ActiveProfiles("otherTestServices")
public class MyOtherTest extends SpringContextTestCase {
@Autowired
private MyService myOtherMockedService;
// ...
}
我有完全相同的问题,我通过使用Mockito.any()
作为参数来解决它
例如:
when(transactionalService.validateProduct(id)).thenReturn("")
=> when(transactionalService.validateProduct(Mockito.any())).thenReturn("")