Mockito when().thenReturn不必要地调用该方法

Krz*_*yna 15 java testing junit mockito

我正在研究一个继承的代码.我写了一个应该捕获NullPointerException的测试(因为它试图从null对象调用一个方法)

@Test(expected=NullPointerException.class)
public void checkXRequirement_NullProduct_AddAction_ShouldThrowNullPointerException() throws CustomException {
  Site site = mock(Site.class);
  Product product = null;
  when(BasketHelper.getAction(request)).thenReturn(0);
  when(BasketHelper.getActionProduct(site, request)).thenReturn(product);
  BasketHelper.requiresX(request, site);

}
Run Code Online (Sandbox Code Playgroud)

相关方法和变量:

public static final int ACTION_ADD = 0;
public static final int ACTION_DELETE = 1;

protected static int getAction(HttpServletRequest a_request) {
  String sBuyProduct = a_request.getParameter(ATTRIBUTE_NAME_BUY_PRODUCT);
  String sBuyProduct = a_request.getParameter(ATTRIBUTE_NAME_BUY_PRODUCT);

  if (sBuyProduct != null) iAction = ACTION_ADD;
  else (sDelProduct != null) iAction = ACTION_DELETE;

  return iBasketAction
}

protected static Product getActionProduct(Site a_site, HttpServletRequest a_request) {

    String sBuyProduct = a_request.getParameter(ATTRIBUTE_NAME_BUY_PRODUCT);
    String sDelProduct = a_request.getParameter(ATTRIBUTE_NAME_DEL_PRODUCT);
    String sProduct = null;

    switch (getBasketAction(a_request)) {
        case BASKET_ACTION_ADD:
        sProduct = sBuyProduct;
    break;
        case BASKET_ACTION_DELETE:
        sProduct = sDelProduct;
    break;
    }

    int iProductId;
    try {
        iProductId = Integer.parseInt(sProduct);
    } catch (NumberFormatException nbrEx) {
        return null;
    }

    Product prod = getProductById(iProductId);

    if (prod.isMasterProduct()) {
        prod = getChildProduct(prod, a_site, a_request);
    }

    return prod;
}


public static boolean requiresX(HttpServletRequest request, Site site) throws CustomException {
  try{
    if (getAction(request) == ACTION_ADD) { 
    Product prod = getActionProduct(site, request);
    return prod.getType().isRequiredX();
    }  
  } catch(NullPointerException exception) {
    log.error("Error Message", exception);
  }
  return false;
}
Run Code Online (Sandbox Code Playgroud)

运行测试的jUnit结果是堆栈跟踪失败:

java.lang.Exception: Unexpected exception, expected<java.lang.NullPointerException> but was<org.mockito.exceptions.misusing.WrongTypeOfReturnValue>
Caused by: org.mockito.exceptions.misusing.WrongTypeOfReturnValue: 
Integer cannot be returned by getParameter()
getParameter() should return String#
Run Code Online (Sandbox Code Playgroud)

我是否误解了何时().thenReturn应该在这里工作?我只是希望getAction返回0并且getActionProduct在被调用时返回null.很明显,getParameter()被调用,我不知道为什么.

KKK*_*der 14

Mockito不能模拟静态方法.您的检查无效时:

  when(BasketHelper.getAction(request)).thenReturn(0);
  when(BasketHelper.getActionProduct(site, request)).thenReturn(product);
Run Code Online (Sandbox Code Playgroud)

这是我们想要减少静态方法使用的另一个原因,因为它很难模拟.

如果您的班级保持这样,就没有更简单的方法来模仿行为.但是,如果您想要更改设计并使两种方法都是非静态的.使用"when"的正确方法是对mocked对象应用检查.例如:

  BasketHelper basketHelper = mock(BasketHelper.class);
  when(basketHelper.getAction(request)).thenReturn(0);
  when(basketHelper.getActionProduct(site, request)).thenReturn(product);
Run Code Online (Sandbox Code Playgroud)

但是,如果您将类的getAction和getProduct方法重新设计为NON-STATIC,这只会再次起作用.

我记得有一些其他测试框架支持模拟静态方法.

  • 我认为PowerMock允许模拟静态方法.也许我应该调查一下.谢谢 (3认同)
  • 也不要总是依靠像PowerMock这样的工具来"强制"进行单元测试.如果您需要这样的特殊工具,那么可能有一种方法可以改进您的设计.换句话说,尝试重构单元测试,以便您可以在没有PowerMock的情况下进行测试.如果您无法这样做(由于某些限制),请随意使用它.外卖 - 只是因为PowerMock或任何类似的工具存在并不会给你一个难以编写测试代码的借口!:) (2认同)