如何进行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
(这种想法有时称为“不要嘲笑您不拥有的类型”)