1

我有一个关于在控制器测试中创建 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 有什么区别?

感谢您的关注!对不起,如果我的问题很愚蠢。我很新,它有效,但我还没有魔法。

4

1 回答 1

2

文档@InjectMocks

Mockito 将尝试仅通过构造函数注入、setter 注入或属性注入按顺序注入模拟

因此,由于EntryService是您的控制器的依赖项,因此@InjectMocks将尝试EntryService在您的测试类中找到一个模拟对象并将其注入mainController.

请注意,只会constructor injection, setter injection, or property injection发生其中之一。

@Mock 将字段标记为模拟对象。@InjectMocks 将模拟对象注入标记的字段,但标记的字段不是模拟。

于 2017-12-30T20:23:54.493 回答