如何进行Android单元测试和模拟静态方法


问题内容

嗨,我真的希望您能为我提供帮助,我感觉我已经把头发拉了好几天了。

我正在尝试为方法A编写单元测试。方法A调用静态方法B。我想模拟静态方法B。

我知道以前已经有人问过这个问题,但是我觉得Android从那时起已经成熟,并且必须有一种方法可以执行如此简单的任务,而无需重写我要测试的方法。

这是一个示例,首先是我要测试的方法:

public String getUserName(Context context, HelperUtils helper) {
    if(helper == null){
        helper = new HelperUtils();
    }
    int currentUserId = helper.fetchUsernameFromInternet(context);

    if (currentUserId == 1) {
        return "Bob";
    } else {
        return "Unknown";
    }
}

接下来,我要模拟的静态方法:

public class HelperUtils {
    public static int fetchUsernameFromInternet(Context context) {
        int userid = 0;

        Log.i("HelperUtils ", "hello");

        return userid;
    }
}

在其他语言中,这是如此简单,但我无法使其在Android中工作。我已经尝试过Mockito,但似乎不支持静态方法

HelperUtils helper = Mockito.mock(HelperUtils.class);
Mockito.when(helper.fetchUsernameFromInternet(getContext())).thenReturn(1);

这个错误

org.mockito.exceptions.misusing.MissingMethodInvocationException

我已经尝试过Powermock,但是我不能完全确定Android是否支持此功能。我设法在gradle文件中使用androidCompile使powermock运行,但是出现以下错误:

错误:任务’:app:dexDebugAndroidTest’的执行失败。com.android.ide.common.process.ProcessException:

更不用说PowerMockito.mockStatic(HelperUtils.class);什么也不返回,所以我不知道该如何传递给我的getUsername方法!

任何帮助将非常感谢。


问题答案:

静态方法与任何对象都不相关-
您的对象与对象helper.fetchUsernameFromInternet(...)相同(但有点混乱)HelperUtils.fetchUsernameFromInternet(...)-因此,您甚至应该收到编译器警告helper.fetchUsernameFromInternet

而且,不必Mockito.mock模拟静态方法,而必须使用:@RunWith(...)@PrepareForTest(...)然后PowerMockito.mockStatic(...)在这里有完整的示例:PowerMockito模拟单个静态方法并返回对象

换句话说,模拟静态方法(以及构造函数)有些棘手。更好的解决方案是:

  • 如果可以更改HelperUtils,请将该方法设置为非静态,然后可以HelperUtils使用通常的方法进行模拟Mockito.mock

  • 如果您不能更改HelperUtils,请创建一个包装类,该包装类委派给原始对象HelperUtils,但没有static方法,然后使用常规方法Mockito.mock(这种想法有时称为“不要嘲笑您不拥有的类型”)