Android 开发 单元测试 (Mock)

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就是创建一个类的虚假的对象,在测试环境中,用来替换掉真实的对象

结果

  1. 用来验证这个对象的某些方法的调用情况,调用了多少次,参数是什么等,
  2. 指定这个对象的某些方法的行为,返回特定的值,或者是执行特定的动作

Mock框架 : Mockito

Mockito是mocking框架,它让你用简洁的API做测试。而且Mockito简单易学,它可读性强和验证语法简洁。
github : https://github.com/mockito/mockito
官网 : http://site.mockito.org/
官方文档 https://static.javadoc.io/org.mockito/mockito-core/2.12.0/org/mockito/Mockito.html

首先还是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()函数。当需要额外的灵活性时你可能需要使用参数匹配器.参数匹配器使验证和测试桩变得更灵活。

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


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(); // 调用会有异常
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()) + "");

    }
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<JUnitTest.NetworkCallback>() {
            @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");
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 210,978评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 89,954评论 2 384
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,623评论 0 345
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,324评论 1 282
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,390评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,741评论 1 289
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,892评论 3 405
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,655评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,104评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,451评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,569评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,254评论 4 328
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,834评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,725评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,950评论 1 264
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,260评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,446评论 2 348