背景

Mockito的强大就不多说明了,这是Mockito2的官方文档:点击这里查看,但有些场景依然无法满足,这时候可以借助powermock,powermock是对Mockito的一些能力的扩展,如支持静态方法 ,私有方法等等场景的mock。PowerMock内部使用了Mockito的API,可以保证同时使用两种mock框架。

现有类BeanUtils如下方法代码:

public String getName() {
//其它逻辑
        String name = new BeanObject().getName();
        return name;
    }

我需要测试BeanUtils类getName()接口,只需确认其中逻辑符合预想,最后调用new BeanObject().getName()的方法的返回值并不关心,也可能BeanObject的getName方法需要传入其它参数,或者这个方法内部有更复杂的逻辑需要回避。

那么我需要mock BeanObejct类的getName方法。

解决方法是mock (BeanObject的mock实例的getName)方法。那这样需要保证每次new BeanObject()的时候构造的都是我mock的那个BeanObject实例。

实现代码

就详细的说下测试实现吧。

1. 先引入测试相关的maven依赖

    <dependencies>
     <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.mockito</groupId>
            <artifactId>mockito-core</artifactId>
            <version>2.8.9</version>
            <scope>test</scope>
        </dependency>

        <!-- https://mvnrepository.com/artifact/org.powermock/powermock-module-junit4 -->
        <dependency>
            <groupId>org.powermock</groupId>
            <artifactId>powermock-module-junit4</artifactId>
            <version>1.7.4</version>
            <scope>test</scope>
        </dependency>

        <!-- https://mvnrepository.com/artifact/org.powermock/powermock-api-mockito2 -->
        <dependency>
            <groupId>org.powermock</groupId>
            <artifactId>powermock-api-mockito2</artifactId>
            <version>1.7.4</version>
            <scope>test</scope>
        </dependency>
   </dependencies>

2. 贴下这几个类的代码

public class BeanUtils {

    public String getName() {
        String name = new BeanObject().getName();
        return name;
    }
}
public class BeanObject {

    private String name = "test_bean";

    public String getName() {
        return name;
    }

}
@RunWith(PowerMockRunner.class)
@PrepareForTest({BeanUtils.class})
public class TestBeanUtils {

    @Test
    public void testGetName() throws Exception {

        BeanObject beanObject = Mockito.mock(BeanObject.class);
        PowerMockito.whenNew(BeanObject.class).withNoArguments().thenReturn(beanObject);
        Mockito.when(beanObject.getName()).thenReturn("mock_test");
        System.out.println(new BeanUtils().getName());
    }
}
PowerMock的使用

借助上面代码TestBeanUtils简单说下配置:

1. 首先在测试类上使用注解@RunWith(PowerMockRunner.class)

2. 使用注解@PrepareForTest.(这个注解也可作用在方法,取决于你想要的作用范围 )

@PrepareForTest这个注解是告诉PowerMock框架,我们需要对测试类作字节码级的处理。这些类可以是final、私有、静态等方法 或者实例的时候返回mock对象。注意我这里声明要处理的是BeanUtils类,是需要测试的类而不BeanObject类。

要mock BeanObject的构造方法,我们可以使用PowerMockito的whenNew(),这是个链式调用,之后调用withNoArguments,见名知义,就是无参构造,当然还有withAnyArguments()、withArguments(...)等根据实际需要,然后返回的使用mock的BeanObject实例。

这样每次new BeanObject的时候,构造出来的都是那个mock的BeanObject实例。

然后,再用下面这行代码,mock它的getName方法,便达到预期结果了。

Mockito.when(beanObject.getName()).thenReturn("mock_test");