junit框架相关源代码分析,网上已经有很多了,本篇不做过多相关解说,主要还是要自己多读相关源代码。本篇主要对自动化测试过程相关的测试用例,测试数据,测试结果结合junit做相关扩展说明。

如果要解读junit详细源码,请参考深入JUnit源码之RunnerJUnit之参数化测试和成组测试的使用相关系列文档,本篇只对主要类进行说明,主要相关类FrameworkMethod,Description,TestClass,BlockJUnit4ClassRunner,Suite

,BlockJUnit4ClassRunnerWithParameters,parameterized,相关关系可见如下图

  1. junit4主要类介绍

Description类主要是对被测试对象的描述,它是一个树型结构;junit中测试对象类和方法,如果Description描述的是一个类,那么它的叶子节点必不为空,如果其描述的为一个方法,则其叶子节点为空;

privatefinal ArrayList<Description> fChildren= newArrayList<Description>(); //无元素

privatefinal String fDisplayName;

privatefinal Annotation[] fAnnotations;

测试树的一般元素/Composite,用fChildren保存其子结点。如一个单元测试类的Description,子结点包括所有@test修饰的方法;而一个Suite的子结点包括几个单元测试类的Description,而单元测试类和成组测试类又可以构成更大的Composite。

FrameworkMethod是对测试方法的封装,描述了执行最终测试的方法。

TestClass是对测试类封装,他只有一个参数Class<?> cls,即被测试的执行的类。

BlockJUnit4ClassRunner类是最基本的Runner,他将传入的Class<?> cls中每个Test注解类封装成一个FrameworkMethod方法并运行。

Suite类可以运行多组不同类型的Runner,比如BlockJUnit4ClassRunner,parameterized或其他自定义Runner;具体见图

parameterized是参数化的Runner,传入的Class<?> cls必须定义一个被Parameters  注解且返回类型为Collection的静态方法,如 :

@Parameters

public static Collection usernameData()

它会先执行@Parameters方法获取集合数据,对集合中的每组数据和Class<?>cls构造一个BlockJUnit4ClassRunnerWithParameters并运行其Class<?> cls中@Test注解方法,一般Class<?> cls的@Test注解方法只有一个。

BlockJUnit4ClassRunnerWithParameters和BlockJUnit4ClassRunner功能差不多,多一个参数Object接受parameterized中集合数据,主要配合Parameterized Runner使用。

 2   junit4中过滤器Filter

BlockJUnit4ClassRunner过滤调用顺序

parameterized过滤调用顺序

   3  junit相关功能扩展设想

        BlockJUnit4ClassRunner是对类中的每个Test注解方法封装成一个FrameworkMethod并运行,比如

         

public class CalculatorTest {

@Test
    public void testAdd() {
        calculator.add(2);
        calculator.add(3);
        assertEquals(5,calculator.getResult());//第一个参数填写期待结果,第二个参数填写实际结果
       
    }

@Test
    public void testSubstract() {
        calculator.add(10);
        calculator.substract(2);
        assertEquals(8,calculator.getResult());
    }

@Test
    public void testMultiply() {
        calculator.add(2);
        calculator.multiply(3);
        assertEquals(6,calculator.getResult());
    }

}

那么testAdd,testSubstract,testMultiply方法都被封装成FrameworkMethod对象,并反射调用执行。其每个方法被设计

成一个用例,方便测试设计,其主要问题:

a.   测试数据被写在方法中,修改和查看都不方便。

b.   一个类里面一个方法只对应一组测试数据(即一个测试用例),这样有很大的代码冗余。

parameterized中的TestCase类必须有@Parameters注解方法,其运行逻辑为,先调用TestCase类@Parameters注解方法,

得到Collection数据,根据Collection和TestCase生成一组BlockJUnit4ClassRunnerWithParameters(带参数)类,对每组BlockJUnit4ClassRunnerWithParameters中TestCase@Test注解方法封装成FrameworkMethod类并运行,这样的场景一般@Test方法只有一个,即使多组数据对应到一个测试方法上组成多组用例,解决上面的b问题。例如:

@RunWith(Parameterized.class)
public class SquareTest {
    private static Calculator calculator = new Calculator();
    private int param;
    private int result;

@Parameters
    public static Collection date(){
        return Arrays.asList(new Object[][]{
            {2, 4},
            {0, 0},
            {-3, 9},
      });
    }

public SquareTest(int param, int result) {
           this.param = param;
           this.result = result;
    }
   
    @Test
    public void square() {
        calculator.square(param);
        assertEquals(result, calculator.getResult());
    }
   
}

由上可知parameterized解决了一个方法对应一组测试数据的问题,使多组数据对应到一个方法上组成多组测试用例(即上面的b问题)。虽然parameterized解决了一定程度上冗余问题,但junit在使用上依然存在可改进,除了之前的a问题(测试数据写在代码里面)外,比如一个接口测试,设计了一组测试数据(如a,b,c,d,e,f,g,h,i,g,k),其中a,b,c组合数据测试代码对应该一个方法,d,e,f,g数据的测试代码对应一个方法,h数据的测试代码单独对应一个方法,i数据测试代码对应一个方法,g,k对应一个方法,按照目前junit设计的BlockJUnit4ClassRunner和parameterized来组织用例代码,那么我需要设计3个parameterized的测试类对应3组测试数据,1个BlockJUnit4ClassRunner的测试类来对应h,i的测试数据,这样就让测试代码比较乱,我们可不可以设计一个测试类,该测试类中有5个方法来运行该组测试数据,这样一个测试类对应一个接口测试,是不是更好一点。

扩展思路,1  改写parameterized,增加Collection 数据可构造传入

2   增加注解@TestCase,其中cstype = 1,rule = {1},支持用例ID配置规则。

3   增加一个过滤器,根据FrameworkMethod中的@TestCase注解和用例ID进行过滤。

增加测试数据读取类,将测试数据构造成Collection 和TestClass  传入改写后的parameterized,对Collection中的每组用例数据,按用例ID与TestClass的所有方法进行匹配,只有@TestCase规则符合用例ID的方法才被执行。见图

在下篇中再做相关代码实现分享

浅析junit4及扩展实践的更多相关文章

  1. Junit4.x扩展:运行指定方法

    相信很多道友搞开发的一般都会用到Junit单元测试工具,不知道大家有没有遇到一个这样的问题: 有的单元测试用例有很多@Test方法,甚至有的方法会执行很长时间,只能空等执行.而实际上我们只需要运行其中 ...

  2. 百万用户时尚分享网站feed系统扩展实践

    Fashiolista是一个在线的时尚交流网站,用户可以在上面建立自己的档案,和他人分享自己的以及在浏览网页时看到的时尚物品.目前,Fashiolista的用户来自于全球100多个国家,用户达百万级, ...

  3. Mac下使用Pecl安装PHP的Swoole扩展实践

    一.背景 前段时间把Mac系统重装了,PHP的一些扩展都没了,昨天需要调试一个swoole开发的项目,发现命令行中的PHP是系统自带的,如果安装swoole扩展很不方便:需要自己手动去下载swoole ...

  4. Scala 中方法扩展实践

    前言 这个名字不知道取得是否合适,简单来说要干的事情就是给某个类型添加一些扩展方法,此场景在各种语言中都会用到,比如 C# 语言,如果我们使用一个别人写好的类库,而又想给某个类库添加一些自己封装的方法 ...

  5. redis水平扩展实践,完全配置,无需代码改动

    设计思路 思路很简单,就是基于用户ID进行分库,将用户的ID字符串按照byte逐个计算ID对应的hash原值(一个数字,取绝对值,因为原始值可能过大溢出,变成负数),然后,再用这个hash原值对库的个 ...

  6. php yaf框架扩展实践五——数据层

    从狭义角度上来理解数据层就是数据库,比较广义的理解来看数据库.远程数据.文件等都可以看做数据层.项目初期的时候一般单一的数据库就可以了,随着流量的增大就要对数据层做很多的改进,例如增加从库分散读压力, ...

  7. lombok自定义扩展实践

    lombok是一款能够在java代码编译阶段改变代码的插件.比如生成setter和getter方法,生成log类变量等,能够简化一些特定的模版式代码.本文将以实现一个基于特定注解生成日志代码的方式,简 ...

  8. Sentinel Slot扩展实践-流控熔断预警实现

    前言 前几天公司生产环境一个服务由于流量上升触发了 Sentinel 的流控机制,然后用户反馈访问慢,定位发现是 task 定时任务导致,后面 task 优化之后发布,流量恢复正常. 这是一个再正常不 ...

  9. 1.还不会部署高可用的kubernetes集群?看我手把手教你使用二进制部署v1.23.6的K8S集群实践(上)

    公众号关注「WeiyiGeek」 设为「特别关注」,每天带你玩转网络安全运维.应用开发.物联网IOT学习! 本章目录: 0x00 前言简述 0x01 环境准备 主机规划 软件版本 网络规划 0x02 ...

随机推荐

  1. HihoCoder 1053 : 居民迁移 二分+贪心+双指针(好题)

    居民迁移 时间限制:3000ms 单点时限:1000ms 内存限制:256MB 描述 公元2411年,人类开始在地球以外的行星建立居住点.在第1326号殖民星上,N个居住点分布在一条直线上.为了方便描 ...

  2. IOS socket编程--Asyncsocket

    iPhone的标准推荐是CFNetwork 库编程,其封装好的开源库是 cocoa AsyncSocket库,用它来简化CFNetwork的调用,它提供了异步操作 主要特性有: 队列的非阻塞的读和写, ...

  3. LG3391 【模板】文艺平衡树(Splay)

    题意 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 3 4 1 \(n,m ...

  4. VBA的过程及参数详解

    VBA的过程及参数详解 VBA中的过程(Procedure)有两种,一种叫函数(Function),另外一种叫子程序(Subroutine),分别使用Function和Sub关键字.它们都是一个可以获 ...

  5. 10 Things ASP.NET Developers Should Know About Web.config Inheritance and Overrides(转)

    10 Things ASP.NET Developers Should Know About Web.config Inheritance and Overrides Wednesday, Janua ...

  6. 使用妹子UI开发的体验分享

    前阵子看到一个类似bootstrap的前端UI框架,好奇心驱使下,去琢磨了一些,最终决定网站改版用这个UI试试效果: 首页+头部: 投稿页: 现成拷贝过来的评论列表: 总结: 上手难度: (熟悉boo ...

  7. 蓝桥杯 算法训练 ALGO-57 删除多余括号

    算法训练 删除多余括号   时间限制:1.0s   内存限制:512.0MB 问题描述 从键盘输入一个含有括号的四则运算表达式,要求去掉可能含有的多余的括号,结果要保持原表达式中变量和运算符的相对位置 ...

  8. 庖丁解牛-----Live555源码彻底解密(根据MediaServer讲解Rtsp的建立过程)

    live555MediaServer.cpp服务端源码讲解 int main(int argc, char** argv) { // Begin by setting up our usage env ...

  9. Java开发进阶技能(附文章引用链接)

    一.玩转源码 1.Java+Selenium3方法篇0-如何在Eclipse上查看Selenium源码 (在github上下载源码)

  10. 第十三章 MySQL用户管理(待续)

    ·············