JAVA

1.junit与Mock

2017-06-07  本文已影响614人  岛在深海处
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()与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.

上一篇下一篇

猜你喜欢

热点阅读