单元测试Junit4+mockito+powermock

1.单元测试的意义

  • 梳理思维逻辑
  • 提升编码能力,架构能力和代码质量
  • 基本思想:输入输出,期望值和真实值对比

不容易编写测试代码的代码很可能就是设计上有缺陷或者不完美,基于测试驱动理念,能写出更加符合设计更加优美的代码

1.1 常见单元测试框架

easymock, testNg, mockito

1.2 单元测试的一些原则

  • 一次只关注一个功能函数。开发者编写的一小段代码,用于检验被测代码的一个很小的、很明确的功能是否正确。通常而言,一个单元测试是用于判断某个特定条件(或者场景)下某个特定函数的行为
  • 单元测试覆盖功能范围。主体功能,边界条件,特殊情况的处理
  • 单元测试覆盖率,不追求100%。语句覆盖,分支覆盖,条件覆盖,分支-条件覆盖,条件组合覆盖,路径覆盖
  • 需要练习才能掌握

2.Junit4使用

简介:单元测试框架,内部涵盖了单元测试的主要流程,为单元测试提供了模板

image.png

@BeforeClass @AfterClass类级别,被标注该注解的方法只会执行一次,方法必须为public static void
@Before @After 会多次执行,每个test方法执行时都会先执行@Before,执行完test方法之后再执行@After。

单元测试启动方式

  • IDEA开发平台自带的Run工具(最常用)
  • 使用单元测试类JUnitCore.runClasses(研究源码的可以通过此类作为入口)
  • mvn命令执行时test阶段

2.2 断言(https://hamcrest.org/JavaHamcrest/tutorial)

前期Junit4使用的是自己包内的断言(扩展性不好,无法动态扩充新的断言语句),后来转为使用hamcrest中的断言(org.hamcrest.MatcherAssert)。hamcrest的核心是引入了Matcher机制,核心类org.hamcrest.CoreMatchers和org.hamcrest.Matchers(hamcrest-library包中),另外可以自定义Matcher。
可以分为几大类

  • 一般匹配断言
    1. assertThat(“myValue”, allOf(startsWith(“my”), containsString(“Val”)))
    2. assertThat( “myValue”, anything() )
    3. assertThat( “myValue”, not(“foo”))
  • 字符串匹配
    1. assertThat(“myStringOfNote”, containsString(“ring”))
    2. assertThat(“myStringOfNote”, startsWith(“my”))
  • 数值相关匹配
    1.assertThat(2, greaterThan(1))
    2.assertThat(1.03, is(closeTo(1.0, 0.03)))
  • 集合相关匹配
    1.assertThat(myMap, hasEntry(“bar”, “foo”))
    2.assertThat(Arrays.asList(“foo”, “bar”), hasItem(startsWith(“ba”)))
    3.assertThat(myMap, hasKey(“bar”))
  • Beans相关
    1.hasProperty
    2.samePropertyValuesAs
    3.getPropertyDescriptor

2.3 Runner

@RunWith
@Ignore
@Rule

//同时执行多个测试类
@RunWith(Suite.class)
@Suite.SuiteClasses({
        ParameterTest.class,
        AccountLoginControllerTest.class
})
public class MultipleTestClasses {
}

2.4 Matcher机制(hamcrest)

2.5 Rule,不经常使用

常用场景:类似Before,After等功能,灵活增加或者修改类中测试方法的行为。eg.在多个测试类中都需要连接,关闭数据库;日志输出;变相支持多个RunWith (https://github.com/junit-team/junit4/wiki/Rules)

pom.xml包依赖

        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.13</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.hamcrest</groupId>
            <artifactId>hamcrest-library</artifactId>
            <version>1.3</version>
            <scope>test</scope>
        </dependency>

3.Mockito

重要的思想:理解mock/stub的基本理念,具体要解决的问题。使用最小化的资源验证代码逻辑是否符合预期,减少对数据库的连接,减少对配置文件的读写,摆脱启动时对框架的依赖。

底层原理简单介绍:
关注点:org.mockito.Mockito 该类包含了大量经常使用的函数, org.mockito.ArgumentMatchers该类抽象出来方法的各种参数

3.1框架启动常见的几种方式

1第一种 @RunWith(MockitoJUnitRunner.class)
2第二种 
    @Before
    public void init(){
        MockitoAnnotations.initMocks(this);
    }

3.2常用注解和函数

@Mock(answer = Answers.RETURNS_SMART_NULLS)
when().thenReturn()
when().thenThrow()
when().thenAnswer()
when().thenCallRealMethod()

3.4 spy

会真实执行

        List<String> realList = new ArrayList<>();
        List<String > list = spy(realList);
        list.add("Mockito");
        assertThat(list.get(0), equalTo("Mockito"));

3.5 verify

模拟验证方法是否执行过以及执行过的次数

        doNothing().when(list).clear();
        list.clear();
        verify(list, times(1)).clear();

3.6 answer

如果在方法调用中需要固定的返回值,则应使用thenReturn(…)。 如果需要执行某些操作或需要在运行时计算值,则应使用thenAnswer(…)

    private List<String> list;
    @Before
    public void init(){
        this.list = mock(ArrayList.class);
    }
    @Test
    public void stubbingWithAnswer(){
        when(list.get(anyInt())).thenAnswer(invocationOnMock ->{
           Integer index = invocationOnMock.getArgument(0);
           return String.valueOf(index * 10);
        });
        assertThat(list.get(0), equalTo("0"));
        assertThat(list.get(999), equalTo("9990"));
    }

3.7 pom.xml包依赖

<!-- https://mvnrepository.com/artifact/org.mockito/mockito-core -->
<dependency>
    <groupId>org.mockito</groupId>
    <artifactId>mockito-core</artifactId>
    <version>3.8.25</version>
    <scope>test</scope>
</dependency>

4.Powermock的功能

概念:扩展Mockito的功能,增加mock final, static, 局部变量和private方法
关注点:org.powermock.api.mockito.PowerMockito类
常用的方法:
mockStatic模拟静态方法
whenNew 模拟局部变量时可以使用

注解

@RunWith(PowerMockRunner.class)
@PrepareForTest()

4.1 常用示例

4.1.1 mock局部变量

UserService类中测试queryUserCount方法
 public int queryUserCount(){
        UserDao userDao = new UserDao();
        int remainCnt = 10; //虚拟额外的逻辑
        return userDao.getCount() + remainCnt;
    }

核心测试代码如下:

            UserDao userDao = PowerMockito.mock(UserDao.class);
            PowerMockito.whenNew(UserDao.class).withNoArguments().thenReturn(userDao);
            PowerMockito.when(userDao.getCount()).thenReturn(10);
            UserService service = new UserService();
            int result = service.queryUserCount();
            assertEquals(20, result);

其中PowerMockito.whenNew(UserDao.class).withNoArguments().thenReturn(userDao)会以局部变量对象注入到queryUserCount()方法中,另外这里的userDao是mock出来的,不能用new的方法创建。

4.1.2 模拟静态方法

    public int queryUserCount(){
        return UserDao.getCount(); // 调用的静态getCount方法
    }

核心测试代码如下:

       mockStatic(UserDao.class);
       when(UserDao.getCount()).thenReturn(10);

注意mockStatic传入的是.class对应调用的方法是PowerMockito.mockStatic(Class<?> type, Class<?>... types)

4.1.3final修饰的class

final public class UserDao{}

使用mockito模拟静态类

    @Mock
    UserDao userDao;
    @Test
    public void testQueryUserCountWithMockito() throws Exception {
        MockitoAnnotations.initMocks(this);
        when(userDao.getCount()).thenReturn(10);
        UserService userService = new UserService(userDao);
        int result = userService.queryUserCount();
        assertThat(10, equalTo(result));
    }

会出现

org.mockito.exceptions.base.MockitoException:
Mockito cannot mock/spy because :
 - final class

修改

@RunWith(PowerMockRunner.class)
@PrepareForTest({UserService.class, UserDao.class})

pom.xml中包依赖

        <dependency>
            <groupId>org.powermock</groupId>
            <artifactId>powermock-api-mockito2</artifactId>
            <version>${powermock-version}</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.powermock</groupId>
            <artifactId>powermock-module-junit4</artifactId>
            <version>${powermock-version}</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.powermock</groupId>
            <artifactId>powermock-core</artifactId>
            <version>${powermock-version}</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.powermock</groupId>
            <artifactId>powermock-module-junit4-rule</artifactId>
            <version>${powermock-version}</version>
            <scope>test</scope>
        </dependency>

参考链接:

代码地址(github) https://github.com/ybyao07/learnJavaUTest.git
junit4官网 https://junit.org/junit4/
mockito官网 https://site.mockito.org/
powermock 官网 https://powermock.github.io/
junit4内部依赖(hamcrest) https://hamcrest.org/
mockito内部依赖包 http://objenesis.org
学习网站:https://www.baeldung.com/
教程:https://www.tutorialspoint.com/mockito/mockito_quick_guide.htm

单元测试体系图

junit4需要了解的知识


image.png

mockito,powermock需要了解的知识


image.png

后续

2017年junit5的出现将单元测试框架整个改版,后续发展方向应该是Junit5 + Mockito4(涵盖了mockito-line,可以模拟静态,final等),powermock将会退出历史舞台(3年没人维护了,还有421个issues和17个Pull requests)。
behavior testing(行为测试): concordion, cucumber

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

推荐阅读更多精彩内容