在编写单元测试时 ,必须牢记不要依赖外部组件。 为了避免这种情况,我们使用了模拟框架,对我来说,最容易使用的是Mockito 。
在本文中,我们将看到在Mockito中使用的一种“高级”技术,可以使用Answer接口在模拟方法上返回相同的参数实例。
在本文中,我们将看到在Mockito中使用的一种“高级”技术,可以使用Answer接口在模拟方法上返回相同的参数实例。
假设我们正在为管理Person和Job类的类编写单元测试,并且在操作中它使用DAO类在Person和Job之间插入关系类(M:N)称为PersonJob 。
例如,正在测试的类如下所示:
public PersonJob createPersonJob(Person person, Job job) {.. some job .. PersonJob personJob = new PersonJob(person, job);return this.personJobDao.create(personJob);}
因此,在这种情况下,您似乎需要嘲笑personJobDao 。
让我们创建模拟并记录交互:
public class WhenAJobIsAssignedToPerson {@Testpublic void relationship_should_be_made_persistent() {PersonJobDao personJobDao = mock(PersonJobDao.class);when(personJobDao.create(any(PersonJob.class))).thenReturn(???)PersonJobManager personJobManager = new PersonJobManager();personJobManager.setPersonJobDao(personJobDao);Person person = new Person();Job job = new Job();PersonJob personJob = personJobManager.createPersonJob(person, job);assertThat(personJob.getPerson(), is(person));assertThat(personJob.getJob(), is(job));}}
是的,您可以看到您不知道要返回什么,因为实例是由被测类创建的,并且在测试方法中,您也不知道哪个实例是由createPersonJob方法创建的。 要解决此问题,您需要使用thenAnswer而不是thenReturn方法:
public class WhenAJobIsAssignedToPerson {@Testpublic void relationship_should_be_made_persistent() {PersonJobDao personJobDao = mock(PersonJobDao.class);when(personJobDao.create(any(PersonJob.class))).thenAnswer(new Answer<PersonJob>() {public PersonJob answer(InvocationOnMock invocation)throws Throwable {return (PersonJob) invocation.getArguments()[0];}});PersonJobManager personJobManager = new PersonJobManager();personJobManager.setPersonJobDao(personJobDao);Person person = new Person();Job job = new Job();PersonJob personJob = personJobManager.createPersonJob(person, job);assertThat(personJob.getPerson(), is(person));assertThat(personJob.getJob(), is(job));}}
需要注意的是回答接口需要您实现答案的方法,这在我们的情况下,简单地返回personJobDao的第一个参数(PersonJob实例)。 创建方法。
现在,我们可以安心编写断言,而不必担心返回的实例。
参考:在One Jar To Rule Them All博客中与 JCG合作伙伴 Alex Soto的Mockito进行回答 。
翻译自: https://www.javacodegeeks.com/2012/07/answering-with-mockito.html