1.JUnit:能够直接在PC上执行;AndroidTest:需要依赖Android设备;
2.测试有返回值得方法用Junit,void方法用Mock。
Mock概念:所谓的mock就是创建一个类的虚假的对象,在测试环境中,用来替换掉真实的对象,以达到两大目的:
1)验证这个对象的某些方法的调用情况,调用了多少次,参数是什么等等
2)指定这个对象的某些方法的行为,返回特定的值,或者是执行特定的动作
3.验证方法是否调用,且参数是否正确
Mockito.verify(objectToVerify).methodToVerify(arguments);
其中,objectToVerify和methodToVerify分别是你想要验证的对象和方法。对应上面的例子
4.Mockito.mock()与Mockito.spy()两个方法的区别:
-
mock()方法对象的所有非void方法都将返回默认值:int,long类型返回0,boolean方法返回false,对象方法返回null等等,而void方法将什么都不做。
-
spy()方法对象既能验证方法调用的功能,同时又能调用这个方法的默认实现(mock对象调用默认实现都将返回默认值)
针对这mock()与spy()有返回值得情况举个例子:
//假设目标类的实现是这样的
public class PasswordValidator {
public boolean verifyPassword(String password) {
return "xiaochuang_is_handsome".equals(password);
}
}
测试代码如下:
@Test
public void testSpy() {
//跟创建mock类似,只不过调用的是spy方法,而不是mock方法。spy的用法
PasswordValidator spyValidator = Mockito.spy(PasswordValidator.class);
//PasswordValidator spyValidator = Mockito.mock(PasswordValidator.class);这种情况无论传何值都将返回false
//在默认情况下,spy对象会调用这个类的真实逻辑,并返回相应的返回值,这可以对照上面的真实逻辑
spyValidator.verifyPassword("xiaochuang_is_handsome"); //true
spyValidator.verifyPassword("xiaochuang_is_not_handsome"); //false
//spy对象的方法也可以指定特定的行为
Mockito.when(spyValidator.verifyPassword(anyString())).thenReturn(true);
//同样的,可以验证spy对象的方法调用情况
spyValidator.verifyPassword("xiaochuang_is_handsome");
Mockito.verify(spyValidator).verifyPassword("xiaochuang_is_handsome"); //pass
}
“void方法什么都不做”我的理解是当mock的对象中的void方法中调用了另一个mock对象的方法时,另一个mock对象的方法不会执行
针对mock()与spy()无返回值的情况举个例子:LoginActivity中有个login方法如下:
public void login(View view) {
String userName = mUserName.getText().toString();
String password = mPassword.getText().toString();
mLoginPresenter.loginPresent(userName, password);
}
可以看到LoginActivity中的login()方法调用了LoginPresenter中的loginPresent()方法,而LoginPresenter中的loginPresent方法代码如下:
public class LoginPresenter {
private UserManager mUserManager;
public void loginPresent(String username, String password) {
mUserManager.performLogin(username, password);
}
public void setUserManager(UserManager userManager) { //<==
this.mUserManager = userManager;
}
}
可以看到LoginPresenter中的loginPresent()方法又调用了UserManager的performLogin()方法,现在我们看下测试代码,如下:
@Test
public void login() throws Exception {
LoginActivity loginActivity = Robolectric.setupActivity(LoginActivity.class);
loginActivity.mLoginPresenter = Mockito.spy(LoginPresenter.class);//正确
//loginActivity.mLoginPresenter = Mockito.mock(LoginPresenter.class);//错误
UserManager userManager = Mockito.mock(UserManager.class);
loginActivity.mLoginPresenter.setUserManager(userManager);
((EditText)loginActivity.findViewById(R.id.et_username)).setText("xiaochuang");
((EditText)loginActivity.findViewById(R.id.et_password)).setText("xiaochuang is handsome");
loginActivity.mLoginBtn.performClick();
//verify(loginActivity.mLoginPresenter).login("xiaochuang", "xiaochuang is handsome"); //mock or spy all success;
verify(userManager).performLogin("xiaochuang", "xiaochuang is handsome");//mock error ; spy success
}
如上代码可以看出,当LoginPresenter对象是通过mock()方法获取的话,运行失败,我的理解是mock()方法获取的mLoginPresenter在调用loginPresent()方法时返回了个什么都没做,导致 mUserManager.performLogin(username, password);都没有执行 所以报Actually, there were zero interactions with this mock.
网友评论