我有一个关于在控制器测试中创建 bean 的问题。比如有一个这样的测试
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = {MainApplicationConfiguration.class, JPAConfig.class})
@WebAppConfiguration
public class TestMainController {
private MockMvc mockMvc;
@Before
public void setUp() {
MockitoAnnotations.initMocks(this);
mockMvc = MockMvcBuilders.standaloneSetup(mainController).build();
}
@InjectMocks
private MainController mainController;
@Mock
private EntryService entryService;
@Autowired
DBEntryRepository repository;
@Test
public void testEntryGet() throws Exception {
List<DBEntry> response_data = new ArrayList<>();
response_data.add(new DBEntry(1, 1, "STR", "DATE"));
Mockito.when(entryService.findAllEntries())
.thenReturn(response_data);
MvcResult result = mockMvc.perform(get("/VT/entry/"))
.andExpect(status().isOk()).andReturn();
verify(entryService, times(1)).findAllEntries();
verifyNoMoreInteractions(entryService);
}
}
和映射到的控制器方法
/VT/入口/
@RequestMapping(value = "/entry/", method = RequestMethod.POST)
public ResponseEntity<Void> createEntry(@RequestBody DBEntry entry, UriComponentsBuilder ucBuilder) {
System.out.println("Creating entry " + entry.getNum());
try {
entryService.saveEntry(entry);
entryService.refreshEntryService();
} catch (Exception e) {
e.printStackTrace();
return new ResponseEntity<Void>(HttpStatus.BAD_REQUEST);
}
HttpHeaders headers = new HttpHeaders();
headers.setLocation(ucBuilder.path("/entry/{id}").buildAndExpand(entry.getId()).toUri());
return new ResponseEntity<Void>(headers, HttpStatus.CREATED);
}
EntryService
带有注释的@Service
注释,并且MainApplicationConfiguration.class
是带有@EnableWebMvc
和扫描项目的配置EntryService
。
我想证明这个控制器真的EntryService
在一个真实的应用程序中使用它,并且所有的都是由MainApplicationConfiguration.class
.
问题是:为什么在我的测试执行范围内我的控制器代码中entryService
带有注释?@Mock
不应该只针对那个实例,并且控制器内部应该实例化另一个bean(EntryService),为什么这个注释模拟了该bean的所有出现(在测试范围内)?我在想,我应该编写整个其他上下文网络上下文,而不是在MainApplicationConfiguration.class
内部模拟它并替换当前定义。我非常困惑为什么这个简单的注释会做出这样的事情。
如果有人能理解这种魔力,请说出@InjectMock 和@Mock 有什么区别?
感谢您的关注!对不起,如果我的问题很愚蠢。我很新,它有效,但我还没有魔法。