Mock

初识 Mockito 这个测试框架后,我们要使用 Mock 的属性创建一个被测试类实例时,大概会下面这么纯手工来打造。

假定类 MyService 有一个属性 MyRepository myRepository:

@Repository
public class MyRepository {
 
    public void doSomething() {
        System.out.println("here's dosomething");
    }
 
    public Model findById(Long id) {
        return new Model(id, "Real Repository");
    }
}

@Service
public class MyService {
 
    @Autowired
    private MyRepository myRepository;
 
    public void doSomething() {
        this.myRepository.doSomething();
    }
 
    public Model findById(Long id) {
        return this.myRepository.findById(id);
    }
}

需要构造 MyService 实例时 Mock 内部状态:

MyRepository myRepository = Mockito.mock(MyRepository.class); 
MyService myService = new MyService(myRepository);

如果所有的 Mock 对象全部通过手工来创建,那就不容易体现出 Mockito 的优越性出来。因此对于被测试对象的创建,Mock 属性的注入应该让 @Mock 和 @InjectMocks这两个注解大显身手了。

  • @Mock:创建一个Mock。
  • @InjectMocks:创建一个实例,其余用@Mock(或@Spy)注解创建的mock将被注入到用该实例中。

@Autowird 等方式完成自动注入。在单元测试中,没有启动 spring 框架,此时就需要通过 @ InjectMocks完成依赖注入。@InjectMocks会将带有@Spy 和@Mock 注解的对象尝试注入到被 测试的目标类中。记住下面这两句话即可:

  • Usually when you are unit testing, you shouldn't initialize Spring context. So remove Autowiring.
  • Usually when you do integration testing, you should use real dependencies. So remove mocking.

所以我们可以得出如下代码:

@RunWith(MockitoJUnitRunner.class)
public class UserServiceTest {
 
    @Mock
    private MyRepository myRepository;
 
    @InjectMocks
    private MyService myService;
 
    @Test
    public void testInjectMocks() {
        System.out.println(myService.getMyRepository().getClass());
    }
}

MyService 被标记了 @InjectMocks,在 setUp方法中 执行 MockitoAnnotations.initMocks(this); 的时候,会将标记了 @Mock 或 @Spy 的属性注入到 service 中。MyService 里面的 MyRepository 完全被Mock实例替换,所有的调用都是针对Mock生成类的。

如果我们还有一个MyController如下,需要注入MyService应该怎么解决呢?

@Controller
public class MyController {
 
    @Autowired
    private MyService myService;
 
    public void doSomething() {
        this.myService.doSomething();
    }
 
    public Model findById(Long id) {
        return this.myService.findById(id);
    }
}

如果我用如下的写法:

@RunWith(MockitoJUnitRunner.class)
public class MyControllerTest {
 
    @Mock
    private MyRepository myRepository;
 
    @InjectMocks
    private MyService myService;
 
    @InjectMocks
    private MyController myController;
 
    @Before
    public void setUp() throws Exception {
        Model model = new Model(11L, "AAA");
        doNothing().when(myRepository).doSomething();
        when(myRepository.findById(11L)).thenReturn(model);
    }
 
    @Test
    public void doSomething() throws Exception {
        this.myController.doSomething();
    }
 
    @Test
    public void findById() throws Exception {
        System.out.println(this.myController.findById(11L));
    }
}

使用Mock打桩的为MyRepository,原本以为使用InjectMocks后,MyService会自动注入MyRepository,MyController会自动注入前的MyService,但是结果并不是这样的。MyController无法识别MyService。MyController实例后,没有给myService属性赋值。于是想在MyService上加个@Mock,虽然编译没问题,但是运行起来异常了:

org.mockito.exceptions.base.MockitoException: This combination of annotations is not permitted on a single field:
@Mock and @InjectMocks

所以InjectMocks字段是无法注入其他InjectMocks字段的。所以我们可以考虑使用Spring来做容器管理,修改Test类:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath:beans.xml"})
public class MyControllerTest {
 
    @Mock
    private MyRepository myRepository;
 
    @InjectMocks
    @Autowired
    private MyService myService;
 
    @Autowired
    private MyController myController;
 
    @Before
    public void setUp() throws Exception {
        MockitoAnnotations.initMocks(this);
        Model model = new Model(11L, "AAA");
        doNothing().when(myRepository).doSomething();
        when(myRepository.findById(11L)).thenReturn(model);
    }
 
    @Test
    public void doSomething() throws Exception {
        this.myController.doSomething();
    }
 
    @Test
    public void findById() throws Exception {
        System.out.println(this.myController.findById(11L));
    }
}

其实不借助容器,也可以手动来赋值。在setup方法中做下修改:

@RunWith(MockitoJUnitRunner.class)
public class MyControllerTest {
 
    @Mock
    private MyRepository myRepository;
 
    @InjectMocks
    private MyService myService;
 
    @InjectMocks
    private MyController myController;
 
    @Before
    public void setUp() throws Exception {
        //通过ReflectionTestUtils注入需要的非public字段数据
        ReflectionTestUtils.setField(myController, "myService", myService);
        Model model = new Model(11L, "AAA");
        doNothing().when(myRepository).doSomething();
        when(myRepository.findById(11L)).thenReturn(model);
    }
 
    @Test
    public void doSomething() throws Exception {
        this.myController.doSomething();
    }
 
    @Test
    public void findById() throws Exception {
        System.out.println(this.myController.findById(11L));
    }
}

@RunWith(PowerMockRunner.class)
@RunWith(MockitoJUnitRunner.class)
@PowerMockRunnerDelegate(SpringJUnit4ClassRunner.class)

 

测试方法的入参和方法调用次数

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值