60

我正在使用以下内容。

Powermock-mockito 1.5.12
Mockito 1.95
junit 4.11

这是我的实用程序类

public void InternalUtils {
    public static void sendEmail(String from, String[] to, String msg, String body) {
    }
}

这是被测类的要点:

public class InternalService {
       public void processOrder(Order order) {
           if (order.isSuccessful()) {
               InternalUtils.sendEmail(...);
           }
       }
}

这是测试:

@PrepareForTest({InternalUtils.class})
@RunWith(PowerMockRunner.class)
public class InternalService {
   public void verifyEmailSend() {
        mockStatic(Internalutils.class);
        doNothing().when(InternalUtils, "sendEmail", anyString(), any(String.class), anyString(), anyString());
        Order order = mock(Order.class);
        when(order.isSuccessful()).thenReturn(true);
        InternalService is = new InternalService();

        verifyStatic(times(1));
        is.processOrder(order);
   }
}

上述测试失败。给定的验证方式是none,但是根据code,如果下单成功则必须发送email。

4

2 回答 2

74

如果你在嘲笑这种行为(用类似的东西doNothing()),那么真的不需要打电话给verify*(). 也就是说,这是我重新编写测试方法的尝试:

@PrepareForTest({InternalUtils.class})
@RunWith(PowerMockRunner.class)
public class InternalServiceTest { //Note the renaming of the test class.
   public void testProcessOrder() {
        //Variables
        InternalService is = new InternalService();
        Order order = mock(Order.class);

        //Mock Behavior
        when(order.isSuccessful()).thenReturn(true);
        mockStatic(Internalutils.class);
        doNothing().when(InternalUtils.class); //This is the preferred way
                                               //to mock static void methods.
        InternalUtils.sendEmail(anyString(), anyString(), anyString(), anyString());

        //Execute
        is.processOrder(order);            

        //Verify
        verifyStatic(InternalUtils.class); //Similar to how you mock static methods
                                           //this is how you verify them.
        InternalUtils.sendEmail(anyString(), anyString(), anyString(), anyString());
   }
}

我分为四个部分,以更好地突出正在发生的事情:

1.变量

我选择在这里声明任何实例变量/方法参数/模拟协作者。如果它在多个测试中使用,请考虑将其作为测试类的实例变量。

2. 模拟行为

这是您定义所有模拟行为的地方。在执行被测代码之前,您在此处设置返回值和期望值。一般来说,如果您在此处设置模拟行为,则以后不需要验证该行为。

3. 执行

这里没有什么花哨的;这只是启动正在测试的代码。我喜欢给它自己的部分来引起人们的注意。

4. 验证

verify这是当您调用任何以or开头的方法时assert。测试结束后,您检查您希望发生的事情是否确实发生了。这是我在您的测试方法中看到的最大错误;你试图在它有机会运行之前验证方法调用。其次是您从未指定哪个要验证

补充说明

这主要是我个人的喜好。您需要按照一定的顺序做事,但在每个分组中都有一点回旋余地。这有助于我快速区分出发生在哪里。

我还强烈建议您浏览以下站点上的示例,因为它们非常强大,可以帮助您处理大多数需要的案例:

于 2013-08-27T13:46:32.707 回答
23

您上述答案已被广泛接受并有据可查,我找到了在此处发布答案的一些原因:-

    doNothing().when(InternalUtils.class); //This is the preferred way
                                           //to mock static void methods.
    InternalUtils.sendEmail(anyString(), anyString(), anyString(), anyString());

在这里,我不明白为什么我们自己调用 InternalUtils.sendEmail。我将在我的代码中解释为什么我们不需要这样做。

mockStatic(Internalutils.class);

所以,我们嘲笑了这门课。现在,让我们看看我们需要如何验证 sendEmail(/..../) 方法。

@PrepareForTest({InternalService.InternalUtils.class})
@RunWith(PowerMockRunner.class)
public class InternalServiceTest {

    @Mock
    private InternalService.Order order;

    private InternalService internalService;

    @Before
    public void setup() {
        MockitoAnnotations.initMocks(this);
        internalService = new InternalService();
    }

    @Test
    public void processOrder() throws Exception {

        Mockito.when(order.isSuccessful()).thenReturn(true);
        PowerMockito.mockStatic(InternalService.InternalUtils.class);

        internalService.processOrder(order);

        PowerMockito.verifyStatic(times(1));
        InternalService.InternalUtils.sendEmail(anyString(), any(String[].class), anyString(), anyString());
    }

}

这两行是神奇之处,第一行告诉 PowerMockito 框架它需要验证它静态模拟的类。但是它需要验证哪种方法?第二行告诉它需要验证哪种方法。

PowerMockito.verifyStatic(times(1));
InternalService.InternalUtils.sendEmail(anyString(), any(String[].class), anyString(), anyString());

这是我班的代码,sendEmail api 两次。

public class InternalService {

    public void processOrder(Order order) {
        if (order.isSuccessful()) {
            InternalUtils.sendEmail("", new String[1], "", "");
            InternalUtils.sendEmail("", new String[1], "", "");
        }
    }

    public static class InternalUtils{

        public static void sendEmail(String from, String[]  to, String msg, String body){

        }

    }

    public class Order{

        public boolean isSuccessful(){
            return true;
        }

    }

}

因为它调用了两次,所以您只需要更改 verify(times(2))... 就是这样。

于 2017-08-16T09:55:30.413 回答