android 测试 mock,Android 开发 单元测试 (Mock)

本文介绍了在Android开发中进行单元测试时如何使用Mock技术。通过Mockito框架,详细讲解了如何验证方法调用、设置测试桩、参数匹配器、验证执行顺序以及为回调打桩等,帮助开发者更好地进行单元测试。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Android 开发 单元测试 (Mock)

上一篇文章中说了JUnit4的作为单元测试的情况下去测试java代码的基本用法, 主要讲到了一些用在有返回值的方法,那么这章, 来介绍怎么测试返回值类型为void的代码的如何测试

假设有这样一段代码

public void loginApp(String name, String password){

if (name == null || name.length() == 0) return;

if (password == null || password.length() < 6) return;

ligon(name, password, new NetworkCallback() {

@Override

public void onSuccess(String msg) {

}

@Override

public void onFailure(String msg) {

}

});

}

public void ligon(String name, String password,NetworkCallback networkCallback){

user.setName(name);

user.setPassword(password);

if (name.equals("TaioPi") && password.equals("123456")) {

networkCallback.onSuccess("OK");

}else {

networkCallback.onFailure("FAIL");

}

}

怎么办

使用Mock(模拟对象)

Mock: Mock的概念,其实很简单, 所谓的mock就是创建一个类的虚假的对象,在测试环境中,用来替换掉真实的对象

结果

用来验证这个对象的某些方法的调用情况,调用了多少次,参数是什么等,

指定这个对象的某些方法的行为,返回特定的值,或者是执行特定的动作

Mock框架 : Mockito

首先还是Mockito 使用的步骤

先导包

testCompile "org.mockito:mockito-core:+"

androidTestCompile "org.mockito:mockito-android:+"

模拟并替换测试代码中外部依赖

执行测试代码

验证测试代码是否被正确的执行

简单实用Mockito

我们这里使用 2.x 来介绍 ,跟刚上一篇一样, 我们这里先把 Mockito的主要功能先列出来, 之后再进行实例使用

1.验证某些行为

一旦mock对象被创建了,mock对象会记住所有的交互。然后你就选择性的验证交互。

// mock creation 创建mock对象

MockotiTestBean mockotiTestBean = mock(MockotiTestBean.class);

//using mock object 使用mock对象

mockotiTestBean.setMockName("TaioPi");

mockotiTestBean.setMockAge("26");

//verification 验证

verify(mockotiTestBean).setMockName("TaioPi");

verify(mockotiTestBean).setMockAge("26");

2.测试桩 Stub

先说一下概念 : Stub完全是模拟一个外部依赖,用来提供测试时所需要的测试数据。

// 你可以mock具体的类型,不仅只是接口

// mockotiTestBean = mock(MockotiTestBean.class);

// stubbing 测试桩

when(mockotiTestBean.getMockName()).thenReturn("TiaoPi");

when(mockotiTestBean.getMockAge()).thenReturn(26);

mockotiTestBean.getMockName();

mockotiTestBean.getMockAge();

mockotiTestBean.getMockAge();

// 验证被调用的次数

verify(mockotiTestBean,times(1)).getMockName();

verify(mockotiTestBean,times(2)).getMockAge();

重点介绍一下打桩

桩,或称桩代码,是指用来代替关联代码或者未实现代码的代码。

打桩的目的

打桩的目的主要有:隔离、补齐、控制。

隔离是指将测试任务从产品项目中分离出来,使之能够独立编译、链接,并独立运行。

补齐是指用桩来代替未实现的代码.

控制是指在测试时,人为设定相关代码的行为,使之符合测试需求。

3.参数匹配器matchers

Mockito以自然的java风格来验证参数值: 使用equals()函数。当需要额外的灵活性时你可能需要使用参数匹配器.参数匹配器使验证和测试桩变得更灵活。

内置参数匹配器 ,,自定义参数匹配器在后面

bc0c5b4fb674

a690f684-07cd-4f8a-b9fd-310b0dedc23d.png

看代码

// stubbing 测试桩

when(mockotiTestBean.getAge(anyInt())).thenReturn(26);

mockotiTestBean.getAge(anyInt())

// 验证被调用的次数

verify(mockotiTestBean,times(1)).getAge(anyInt());

4.验证函数的确切,最少,从未调用的次数

mockotiTestBean.setMockName("TiaoPi");

mockotiTestBean.setMockName("TiaoPi");

mockotiTestBean.setMockName("XiaoTiaoPi");

mockotiTestBean.setMockAge(26);

mockotiTestBean.setMockAge(26);

mockotiTestBean.setMockAge(26);

mockotiTestBean.setMockAge(26);

mockotiTestBean.setMockAge(26);

//验证调用

verify(mockotiTestBean,times(2)).setMockName("TiaoPi");

verify(mockotiTestBean,times(1)).setMockName("XiaoTiaoPi");

verify(mockotiTestBean,times(5)).setMockAge(26);

// 使用never()进行验证,never相当于times(0)

verify(mockotiTestBean, never()).setMockName("000");

// 使用atLeast()/atMost()

verify(mockotiTestBean, atLeastOnce()).setMockName("XiaoTiaoPi");

verify(mockotiTestBean, atLeast(2)).setMockName("TiaoPi");

verify(mockotiTestBean, atMost(5)).setMockAge(26);

5.为返回值为void的海曙通过打桩抛出异常

stubVoid(Object) 函数用于为无返回值的函数打桩。现在stubVoid()函数已经过时,doThrow(Throwable)成为了它的继承者

doThrow(new RuntimeException("调用了!!!"))

.when(mockotiTestBean).getVoid();

mockotiTestBean.getVoid(); // 调用会有异常

bc0c5b4fb674

8778512d-20c6-4af9-b902-4706e5741438.png

6.验证执行顺序

创建InOrder对象 : 对那些需要验证顺序的mock对象来创建InOrder对象。

验证执行顺序是非常灵活的-你不需要一个一个的验证所有交互,只需要验证你感兴趣的对象即可。

mockotiTestBean.setMockName("TiaoPi1");

mockotiTestBean.setMockName("TiaoPi2");

// 为该mock对象创建一个inOrder对象 //多个mock 需要创建多个

InOrder inOrder = inOrder(mockotiTestBean);

//确保函数首先执行的是setMockName("TiaoPi1"),然后才是setMockName("TiaoPi2");

inOrder.verify(mockotiTestBean).setMockName("TiaoPi1");

inOrder.verify(mockotiTestBean).setMockName("TiaoPi2");

7.确保交互操作不会执行在mock对象上

确保模拟对象上无互动发生

mockotiTestBean.setMockName("TiaoPi");

// 普通验证

verify(mockotiTestBean).setMockName("TiaoPi");

//验证这个交互从来没有执行

verify(mockotiTestBean,never()).setMockName("xiaoTiaoPi");

MockotiTestBean mockotiTestBean2 = mock(MockotiTestBean.class);

// mockotiTestBean2.getVoid(); //如果调用 就回出异常 因为发生过交互

//验证mock对象从来没有交互

verifyZeroInteractions(mockotiTestBean2);

8.找出冗余的互动

未被验证到的Mock

verifyNoMoreInteractions()并不建议在每个测试函数中都使用 ,尽量实用never()

mockotiTestBean.setMockName("TiaoPi");

mockotiTestBean.setMockAge(26);

verify(mockotiTestBean).setMockName("TiaoPi");

verify(mockotiTestBean).setMockAge(26);

//setMockName 和 setMockAge 都需要验证 否则 下面的验证将会失败

verifyNoMoreInteractions(mockotiTestBean);

9. 简化mock对象的创建

最小化重复的创建代码

使测试类的代码可读性更高

使验证错误更易于阅读,因为字段名可用于标识mock对象

使用@Mock 注解和 MockitoAnnotations.initMocks(this)

注意: MockitoAnnotations.initMocks(this)需要放在测试执行之前

@Before

public void setup() {

MockitoAnnotations.initMocks(this);

}

/**

* 简化mock对象的创建

*/

@Mock

MockotiTestBean mockotiTestBean2;

@Test

public void testMockCreate(){

mockotiTestBean2.setMockAge(anyInt());

verify(mockotiTestBean2).setMockAge(anyInt());

}

10. 为连续的调用做测试桩 stub

有时我们需要为同一个函数调用的不同的返回值或异常做测试桩。典型的运用就是使用mock迭代器。

/**

* 为连续的调用做测试桩 `stub`

*/

@Test

public void testConsecutive() throws Exception{

//连续调用

when(mockotiTestBean.getAge(anyInt())).thenReturn(26)

.thenReturn(25)

.thenReturn(24);

//简单模式

// when(mockotiTestBean.getAge(anyInt())).thenReturn(26,25,24);

//第一次调用

Log.d("第一次调用",mockotiTestBean.getAge(anyInt()) + "");

Log.d("第二次调用",mockotiTestBean.getAge(anyInt()) + "");

Log.d("第三次调用",mockotiTestBean.getAge(anyInt()) + "");

}

bc0c5b4fb674

d265f3b6-f40c-4a82-9ca7-ed5aea7346f8.png

11. 为回调打桩

建议使用thenReturn() 或thenThrow()来打桩

doThrow(new RuntimeException("调用了Click")).when(mockotiTestBean)

.setMockClick(any(MockotiTestBean.MockClick.class));

mockotiTestBean.setMockClick(new MockotiTestBean.MockClick() {

@Override

public void click(int a) {

}

});

12.doReturn()、doThrow()、doAnswer()、doNothing()、doCallRealMethod()系列方法的运用

通过when(Object)为无返回值的函数打桩有不同的方法,因为编译器不喜欢void函数在括号内…

使用doThrow(Throwable) 替换stubVoid(Object)来为void函数打桩是为了与doAnswer()等函数族保持一致性。

当你调用doThrow(), doAnswer(), doNothing(), doReturn() and doCallRealMethod() 这些函数时可以在适当的位置调用when()函数. 当你需要下面这些功能时这是必须的:

测试void函数

在受监控的对象上测试函数

不知一次的测试为同一个函数,在测试过程中改变mock对象的行为。

最后. 监控真实对象spy

为真实对象创建一个监控(spy)对象;当你使用这个spy对象时真实的对象也会也调用,除非它的函数被stub了。

spy与mock的唯一区别就是默认行为不一样:spy对象的方法默认调用真实的逻辑,mock对象的方法默认什么都不做,或直接返回默认值

MockotiTestBean spy = spy(mockotiTestBean);

//打桩

when(spy.getAge(anyInt())).thenReturn(26);

Log.d("获去getAge",spy.getAge(anyInt()) + "'");

//调用真实对象的函数

spy.setMockName("TaioPi");

verify(spy).setMockName("TaioPi");

介绍完成之后, 我们回到之前的问题, 为上面的代码进行测试

public void loginApp(String name, String password){

if (name == null || name.length() == 0) return;

if (password == null || password.length() < 6) return;

ligon(name, password, new NetworkCallback() {

@Override

public void onSuccess(String msg) {

}

@Override

public void onFailure(String msg) {

}

});

}

public void ligon(String name, String password,NetworkCallback networkCallback){

user.setName(name);

user.setPassword(password);

if (name.equals("TaioPi") && password.equals("123456")) {

networkCallback.onSuccess("OK");

}else {

networkCallback.onFailure("FAIL");

}

}

测试代码

首先验证login()

@Test

public void testLogin () throws Exception{

JUnitTest jUnitTest = mock(JUnitTest.class);

JUnitTest.NetworkCallback networkCallback = mock(JUnitTest.NetworkCallback.class);

doAnswer(new Answer() {

@Override

public JUnitTest.NetworkCallback answer(InvocationOnMock invocation) throws Throwable {

Object[] arguments = invocation.getArguments();

JUnitTest.NetworkCallback networkCallback = (JUnitTest.NetworkCallback) arguments[2];

networkCallback.onFailure("500");

return networkCallback;

}

}).when(jUnitTest).ligon(anyString(),anyString(),any(JUnitTest.NetworkCallback.class));

doNothing().when(networkCallback).onFailure("500");

jUnitTest.ligon("TaiaoPi0000", "123456", networkCallback);

verify(jUnitTest).ligon("TaiaoPi0000","123456", networkCallback);

//验证回调是不是执行了

// doCallRealMethod().when(jUnitTest).ligon(anyString(),anyString(),any(JUnitTest.NetworkCallback.class));

}

然后验证 loginApp()

@Test

public void testLoginApp () throws Exception{

JUnitTest jUnitTest = mock(JUnitTest.class);

//这里什么都不做 执行了就好

doNothing().when(jUnitTest).loginApp(anyString(),anyString());

jUnitTest.loginApp("TaiaoPi","123456");

verify(jUnitTest).loginApp("TaiaoPi","123456");

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值