草庐IT

Junit + Idea 参数化与非参数化白盒单元测试

Evagarden 2023-04-11 原文

在进行单元测试的时候,往往要设计多个用例逐个放入程序中执行并返回结果,如果我们想自动完成循环放入用例的操作就需要自己写循环语句。而Junit4的新特性就可以直接帮你完成循环的操作,你只需要将用例以特定的格式写入即可,这就是参数化测试,但是这存在一个限制,一个测试类只能测试一个方法。
非参数化测试的概念我一直没搜到过,但是根据老师给的实例,大概率就是不使用Junit4的新特性…使用Junit3的标准来实现测试,所以涉及到循环也是自己写的,好处是可以在当前类内为每一个覆盖都单独创建一个测试方法。

JUnit4参数化测试

Junit4可以把你的所有测试用例自动运行一遍,并且自带判断函数(这里叫断言)

idea中自带Junit4,但有个提高效率的插件:JunitGenerater V2.0,这个插件是使用Junit这个工具的工具…可以帮你自动生成测试代码框架,类似于打游戏自动寻路。

操作思路

1、安装插件。
2、写好需要测试的目标函数。
3、通过插件自动生成测试类。
4、将设计好的测试用例填进去,运行。

安装插件

settings -> Plugins 搜索安装,安装完会提示重启idea。

安装完成后需要注意一个配置,默认的template改为Junit4。

测试目标

这里简单看一下我的作业。

使用插件

使用插件的方法就是万能右键 -> Generate… -> Test…
有快捷键 Alt+insert 但是我懒得记。
需要注意的是光标要在函数体范围内右键。

勾选上想要测试的函数名,会自动生成测试函数。
@Before和@After可选,这两个类似于生命周期函数,一个是在test函数执行前必会执行,一个在结束后必会执行。虽然我勾了但不一定用到,看习惯。

三个勾选会生成三个函数,当然你不勾自己手打也是一样的。

完善程序

准备测试数据

测试函数使用的数据来自它所在的测试类,所以要在测试类中定义需要的数据。从我要测试的方法中可以看出来输入为x、y、z三个变量,但是还需要一个期望结果expected让程序判断测试是否通过。

	private int x;
    private int y;
    private int z;
    private int expected;

这些变量的值每过一个测试用例就会变一次,需要一个构造函数来赋值。

	public OriginTest(int x, int y, int z, int expected){
        this.x = x;
        this.y = y;
        this.z = z;
        this.expected = expected;
    }

在一个函数中初始化所有测试用例,这个函数需要org.junit.runners.Parameterized.Parameters的注解、static的修饰、Collection的返回值,其中的数据依次对应构造函数的入参。这里我放入的是语句覆盖的测试用例,比较简短。之后可以手动换成别的覆盖数据,或者考虑通过文件操作从txt获取数据。

	@Parameterized.Parameters
    public static Collection testData(){
        Object[][] object={{0,0,50,0},{12,11,100,24},{4,9,-1,18}};
        return Arrays.asList(object);  // 数组转化为集合
    }

测试函数

调用要测试的函数,然后比较结果是否一致。
assertEquals()用来断言两个值是否相同,当然还存在其他断言函数。

	@org.junit.Test
    public void origin() {
        int result = origin.ori(x, y, z);
        assertEquals(expected, result);
    }

完整代码

import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;

import java.util.Arrays;
import java.util.Collection;

import static org.junit.Assert.*;

@RunWith(Parameterized.class)
public class OriginTest {
    private Origin origin;

    private int x;
    private int y;
    private int z;
    private int expected;

    public OriginTest(int x, int y, int z, int expected){
        this.x = x;
        this.y = y;
        this.z = z;
        this.expected = expected;
    }

    @org.junit.Before
    public void setUp() throws Exception {
        origin = new Origin();
    }

    @org.junit.After
    public void tearDown() throws Exception {
    }

    @Parameterized.Parameters
    public static Collection testData(){
        Object[][] object={{0,0,50,0},{12,11,100,24},{4,9,-1,18}};
        return Arrays.asList(object);
    }

    @org.junit.Test
    public void origin() {
        int result = origin.ori(x, y, z);
        assertEquals(expected, result);
    }
}


输出结果

如果没什么问题会每个打个勾,有错的用例会打岔并且输出期望和实际结果。

JUnit3非参数化测试(自动化测试)

我们前面看到的Junit4创建的测试类,你的所有测试用例数据会依次赋值给这个类里声明的私有变量,测试方法也直接使用类内声明变量,没有办法做到在一个类里设计多个函数来执行不同的逻辑覆盖测试。Junit3就比较简陋,除了可以识别测试方法来运行这个单个方法,没什么别的功能了,但就意味着自由度也高。

操作思路

1、创建Junit3测试类
2、在这个测试类为每种覆盖都创建一个测试方法
3、然后想测试哪个覆盖就测试哪个覆盖

创建测试类

同样是右键 -> Generate… -> test…
唯一的不同是将Junit选择为Junit3

创建完成后只是有一个测试类和测试方法,Junit3不使用注解,它通过方法名来判断该方法是否为一个测试函数,所以之后自己创建测试方法名字必须以test开头。

完善程序

基本的参数循环放入逻辑都需要自己写,由于每个覆盖的测试方法只有测试用例不同,我可以将相同的那部分逻辑放到一个方法内实现。实际结果与预期结果的判断可以自己判断也可以使用assert断言函数,但是junit3中的assert如果出现结果不相符返回fasle会出现报错并中止程序,如果希望程序继续运行就需要捕获异常,捕获的异常类型为AssertionError。

完整代码

import junit.framework.TestCase;


public class OriginTest3 extends TestCase {
    private Origin origin;

    public void runTest(int[][] object){
        origin = new Origin();
        for(int i = 0; i < object.length; i++){
            int result = origin.ori(object[i][0], object[i][1], object[i][2]);
            int number = i+1;
            System.out.print("case"+number+": expect:"+object[i][3]+" actual:"+result+" ");
            if (result == object[i][3]) System.out.println("success");else System.out.print("failure ");
            try{
                assertEquals(result, object[i][3]);
            }catch (AssertionError e){
                System.out.println(e.getMessage());
            }
        }
    }

    //语句覆盖
    public void testStatements() {
        int[][] object={{0,0,50,0},{12,11,100,24},{4,9,-1,18}};
        runTest(object);
    }

    //判定覆盖
    public void testDetermine() {
        int[][] object={{0,0,50,0},{12,11,100,24},{24,12,100,24},{4,9,100,18},{3,9,100,9}};
        runTest(object);
    }

    //条件覆盖
    public void testConditions() {
        int[][] object={{0,0,50,0},{1,12,-1,12},{24,12,100,24},{4,9,100,18},{3,9,100,9}};
        runTest(object);
    }

    //判定条件覆盖
    public void testDc() {
        int[][] object={{0,0,50,0},{13,12,-1,26},{24,12,100,24},{4,9,100,18},{3,9,100,9}};
        runTest(object);
    }

    //条件组合覆盖
    public void testCc() {
        int[][] object={{0,0,50,0},{13,12,-1,26},{13,12,100,26},{24,12,100,24},{5,12,100,24},{10,9,100,18},{4,9,100,18}};
        runTest(object);
    }

    //路径覆盖
    public void testPath() {
        int[][] object={{0,0,50,0},{12,11,100,24},{22,11,100,22},{4,9,100,18},{3,9,100,9}};
        runTest(object);
    }

}

参考文献

安装和使用插件:
https://www.csdn.net/tags/NtjaUg5sNzk2MjctYmxvZwO0O0OO0O0O.html
参数化测试:
https://www.cnblogs.com/softidea/p/4155312.html
https://www.cnblogs.com/byron0918/p/4801152.html

有关Junit + Idea 参数化与非参数化白盒单元测试的更多相关文章

  1. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

    很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

  2. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

    我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

  3. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  4. ruby - Ruby 的 Hash 在比较键时使用哪种相等性测试? - 2

    我有一个围绕一些对象的包装类,我想将这些对象用作散列中的键。包装对象和解包装对象应映射到相同的键。一个简单的例子是这样的:classAattr_reader:xdefinitialize(inner)@inner=innerenddefx;@inner.x;enddef==(other)@inner.x==other.xendenda=A.new(o)#oisjustanyobjectthatallowso.xb=A.new(o)h={a=>5}ph[a]#5ph[b]#nil,shouldbe5ph[o]#nil,shouldbe5我试过==、===、eq?并散列所有无济于事。

  5. ruby - RSpec - 使用测试替身作为 block 参数 - 2

    我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere

  6. ruby - 如何在 Ruby 中拆分参数字符串 Bash 样式? - 2

    我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"

  7. ruby - 检查方法参数的类型 - 2

    我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)

  8. ruby - Sinatra:运行 rspec 测试时记录噪音 - 2

    Sinatra新手;我正在运行一些rspec测试,但在日志中收到了一堆不需要的噪音。如何消除日志中过多的噪音?我仔细检查了环境是否设置为:test,这意味着记录器级别应设置为WARN而不是DEBUG。spec_helper:require"./app"require"sinatra"require"rspec"require"rack/test"require"database_cleaner"require"factory_girl"set:environment,:testFactoryGirl.definition_file_paths=%w{./factories./test/

  9. ruby-on-rails - 在默认方法参数中使用 .reverse_merge 或 .merge - 2

    两者都可以defsetup(options={})options.reverse_merge:size=>25,:velocity=>10end和defsetup(options={}){:size=>25,:velocity=>10}.merge(options)end在方法的参数中分配默认值。问题是:哪个更好?您更愿意使用哪一个?在性能、代码可读性或其他方面有什么不同吗?编辑:我无意中添加了bang(!)...并不是要询问nobang方法与bang方法之间的区别 最佳答案 我倾向于使用reverse_merge方法:option

  10. ruby-on-rails - 迷你测试错误 : "NameError: uninitialized constant" - 2

    我遵循MichaelHartl的“RubyonRails教程:学习Web开发”,并创建了检查用户名和电子邮件长度有效性的测试(名称最多50个字符,电子邮件最多255个字符)。test/helpers/application_helper_test.rb的内容是:require'test_helper'classApplicationHelperTest在运行bundleexecraketest时,所有测试都通过了,但我看到以下消息在最后被标记为错误:ERROR["test_full_title_helper",ApplicationHelperTest,1.820016791]test

随机推荐