• [转载]JUnit3 与 JUnit4 的区别


    Unit3 与 JUnit4 的区别

    标签: junitandroidjava测试
     分类:

    本想总结下JUnit3和JUnit4的区别,方便自己的同时也方便他人,不想却违反了DRY原则,网上已经出现了很多的文章。

    既然无须再重复造轮子,就在此推荐两篇个人认为不错的文章,地址如下:

    360doc:http://www.360doc.com/content/12/0202/23/1542811_183778321.shtml

    百度文库:http://wenku.baidu.com/link?url=AJOP-jcuMM9Zdx_iyyKPpyUBOOu06PhWVk89AWkRVbYShzd8j6713ztqe453ta7yMDNVqjpQWmm7j8YbvqHy-EW-uwfFb4_rf-nW84n9Po3

    将文章内容拷贝如下,以防止原文被删除,无法参阅。

    ---------------- begin -------------------------------

    JUnit 4是与JUnit3完全不同的API,它基于Java 5.0中的注解、静态导入等构建而成。JUnit 4更简单、更丰富、更易于使用,并引入了更为灵活的初始化和清理工作,还有限时的和参数化测试用例。

    代码实例最能说明问题。因此,在本文中,我将使用一个例子来展示不同的测试用例:一个计算器。该示例计算器很简单,效率并不高,甚至还有一些错误; 它仅仅操作整数,并且把结果存储在一个静态变量中。Substract方法并不返回一个有效的结果,而且也没有实现乘法运算,而且看上去在 squareRoot方法中还存在一个错误:无限循环。这些错误将帮助说明使用JUnit 4进行测试的有效性。你可以打开和关闭这个计算器,而且你可以清除这些结果。下面是其实现代码:

    package calc;
    
    public class Calculator {
    
    //存储结果的静态变量
    
    private static int result;
    
    public void add(int n) {
    
    result = result + n;
    
    }
    
    public void substract(int n) {
    
    result = result - 1; //错误:应该是"result = result - n"
    
    }
    
    //还没实现
    
    public void multiply(int n) {
    
    }
    
    public void divide(int n) {
    
    result = result / n;
    
    }
    
    public void square(int n) {
    
    result = n * n;
    
    }
    
    public void squareRoot(int n) {
    
    for (; ;) ;//错误:无限循环
    
    }
    
    //清除结果
    
    public void clear() {
    
    result = 0;
    
    }
    
    //打开屏幕,显示"hello",并报警
    
    public void switchOn() {
    
    result = 0;
    
    //实现其它的计算器功能
    
    }
    
    //显示"bye bye",报警,并关闭屏幕
    
    public void switchOff() {
    
    }
    
    public int getResult() {
    
    return result;
    
    }
    
    }

    以下代码是基于JUnit3.8实现的单元测试:

    package junit3;
    
    import calc.Calculator;
    
    import junit.Framework.TestCase;
    
    public class CalculatorTest extends TestCase {
    
    private static Calculator calculator = new Calculator();
    
    @Override
    
    protected void setUp() {
    
    calculator.clear();
    
    }
    
    public void testAdd() {
    
    calculator.add(1);
    
    calculator.add(1);
    
    assertEquals(calculator.getResult(), 2);
    
    }
    
    public void testSubtract() {
    
    calculator.add(10);
    
    calculator.subtract(2);
    
    assertEquals(calculator.getResult(), 8);
    
    }
    
    public void testDivide() {
    
    calculator.add(8);
    
    calculator.divide(2);
    
    assertTrue(calculator.getResult() == 5);
    
    }
    
    public void testDivideByZero() {
    
    try {
    
    calculator.divide(0);
    
    fail();
    
    } catch(ArithmeticException e) {
    
    }
    
    }
    
    public void notReadyYetTestMultiply() {
    
    calculator.add(10);
    
    calculator.multiply(10);
    
    assertEquals(calculator.getResult(), 100);
    
    }
    
    } 

    以下代码是基于JUnit4实现的单元测试:

    package JUnit 4;
    
    import calc.Calculator;
    
    import org.junit.Before;
    
    import org.junit.Ignore;
    
    import org.junit.Test;
    
    import static org.junit.Assert.*;
    
    public class CalculatorTest {
    
    private static Calculator calculator = new Calculator();
    
    @Before
    
    public void clearCalculator() {
    
    calculator.clear();
    
    }
    
    @Test
    
    public void add() {
    
    calculator.add(1);
    
    calculator.add(1);
    
    assertEquals(calculator.getResult(), 2);
    
    }
    
    @Test
    
    public void subtract() {
    
    calculator.add(10);
    
    calculator.subtract(2);
    
    assertEquals(calculator.getResult(), 8);
    
    }
    
    @Test
    
    public void divide() {
    
    calculator.add(8);
    
    calculator.divide(2);
    
    assert calculator.getResult() == 5;
    
    }
    
    @Test(expected = ArithmeticException.class)
    
    public void divideByZero() {
    
    calculator.divide(0);
    
    }
    
    @Ignore("not ready yet")
    
    @Test
    
    public void multiply() {
    
    calculator.add(10);
    
    calculator.multiply(10);
    
    assertEquals(calculator.getResult(), 100);
    
    }
    
    } 


    JUnit3 和 JUnit4的区别

    1、JUnit 4使用org.junit.*包而JUnit 3.8使用的是junit.Framework.*;为了向后兼容,JUnit4发行版中加入了这两种包。

    2、JUnit3中,测试类需要继承junit.framework.TestCase类,而在JUniy4则不用。

    3、JUnit3通过分析方法名称来识别测试方法:方法名必须以“test”为前缀,它必须返回void,而且它必须没有任何参数(例如 public void testDivide())。不遵循这个命名约定的测试方法将被JUnit框架忽略,而且不抛出任何异常(指示发生了一个错误)。

    在JUnit4中,测试方法不必以'test'为前缀,而是使用@Test注解。但测试方法也必须返回void并且无参。在JUnit4中,可以在运行时控制这个要求,并且不符合要求的话会抛出一个异常:

    java.lang.Exception: Method xxx should have no parameters
    java.lang.Exception: Method xxx should be void

    @Test注解支持可选参数。它声明一个测试方法应该抛出一个异常。如果它不抛出或者如果它抛出一个与事先声明的不同的异常,那么该测试失败。

    4、在JUnit3.8中,TestCase类定义了assertEquals()方法,如果要在JUnit中向后兼容,必须静态地导入Assert类。这样一来,就可以像以前一样使用assertEquals方法。

    另外,在JUnit 4中,还引入了两个新的断言方法,它们专门用于数组对象的比较。如果两个数组包含的元素都相等,那么这两个数组就是相等的。

    public static void assertEquals(String message, Object[] expecteds, Object[] actuals);
    
    public static void assertEquals(Object[] expecteds, Object[] actuals); 

    由于JDK 5.0的自动装箱机制的出现,原先的12个assertEquals方法全部去掉了。

    例如,原先JUnit 3.8中的assertEquals(long,long)方法在JUnit 4中要使用assertEquals(Object,Object)。对于assertEquals(byte,byte)、 assertEquals(int,int)等也是这样。这种改进将有助于避免反模式。

    在JUnit 4中,新集成了一个assert关键字(见我们的例子中的divide()方法)。你可以象使用assertEquals方法一样来使用它,因为它们都抛出相同的异常(java.lang.AssertionError)。JUnit 3.8的assertEquals将抛出一个junit.framework.AssertionFailedError。注意,当使用assert时, 你必须指定Java的"-ea"参数;否则,断言将被忽略。

    5、预设环境(Fixture)

    Fixture是在测试期间初始化和释放任何普通对象的方法。在JUnit 3.8中,你要使用setUp()来实现运行每一个测试前的初始化工作,然后使用tearDown()来进行每个测试后的清理。这两个方法在 TestCase类中都得到重载,因此都被唯一定义。注意,我在这个Setup方法使用的是Java5.0内置的@Override注解-这个注解指示该 方法声明要重载在超类中的方法声明。在JUnit 4中,则代之使用的是@Before和@After注解;而且,可以以任何命名。

    6、如果想忽略某个测试方法,在JUnit3中,通过注释掉该方法或者改变命名约定,这样测试运行机就无法找到它;问题随之而来,如果在众多测试中忽略某些测试方法,你可能记不住重命名这个方法;

    而在JUnit4中,把@Ignore注解添加到@Test的前面或者后面即可。其好处在于,测试运行机将会统计出忽略的测试方法数目、运行的测试方法数目以及运行失败的测试方法数目。@Ignore使用一个可选参数(一个String)记录方法被忽略的原因。

    7、在JUnit3.8中,可以选择使用若干运行机:文本型,AWT或者Swing。JUnit4仅使用文本测试运行机。

    注意,JUnit4不会显示任何绿色条来通知你测试成功了。如果你想看到任何类型的绿色的话,那么你可能需要使用JUnit扩展或一种集成了JUnit的IDE(例如 IDEA或者Eclipse)。


    JUnit4的高级特性

    @BeforeClass/@AfterClass与@Before/@After 注解比较

    @BeforeClass和@AfterClass

    @Before和@After

    在每个类中只有一个方法能被注解。

    多个方法能被注解,但其执行的顺序未特别指定,且不运行重载方法。

    方法名是不相关的。

    方法名是不相关的。

    每个类运行一次。

    在每个测试方法执行前/执行后运行。

    在当前类的@BeforeClass方法运行前先运行超类的@BeforeClass方法。在超类中声明的@AfterClass方法将在所有当前类的该方法运行后才运行。 

    超类中的@Before在所有子类的该方法运行前运行。在超类中的@After在在所有子类的该方法运行后才运行。

    必须是公共和静态的。

    必须是公共和非静态的。

    即使一个@BeforeClass方法抛出一个异常,所有的@AfterClass方法也保证被运行。

    即使一个@Before或者@Test方法抛出一个异常,所有的@After方法也保证被运行。

    关于限时测试

    Test注解的timeout参数可以指定方法执行时间(单位毫秒),如果方法没有在指定时间内结束,则测试不通过。

    关于参数化测试

    当使用不同的参数对同一方法测试时,可以使用”参数化测试”加以优化,可参照相关资料。

    关于测试集

    为了在JUnit 3.8的一个测试集中运行若干测试类,你必须在你的类中添加一个suite()方法。而在JUnit 4中,你可以使用注解来代之。你需要使用@RunWith和@Suite注解编写一个空类。

    package test;
    
    import org.junit.runner.RunWith;
    
    import org.junit.runners.Suite;
    
    @RunWith(Suite.class)
    
    @Suite.SuiteClasses({AAATest.class,BBBTest.class})
    
    public class AllTests {
    
    } 

    在此,@RunWith注解告诉JUnit它使用org.junit.runner.Suite。这个运行机允许你手工地构建一个包含测试(可能来自许多类)的测试集。这些类的名称都被定义在@Suite.SuiteClass中。当你运行这个类时,它将运行AAATest和BBBTest。

    关于测试运行机

    JUnit4中广泛地使用了测试运行机。如果没有指定@RunWith,仍会使用一个默认的运行机(org.junit.internal.runners.TestClassRunner)执行。带有@Test的方法的类都隐含地拥有一个@RunWith。

     转自: http://blog.csdn.net/zhang103886108/article/details/45169561

  • 相关阅读:
    Redis 3.2 版本后 list 的实现
    每当发生一次垃圾收集,所有用户线程都必须跑到最近的一个安全点然后挂起线程来等待垃圾回收
    你了解dt.jar吗
    spring boot + vue + element-ui全栈开发入门——开篇
    玩转spring boot——开篇
    java结合node.js非对称加密,实现密文登录传参——让前后端分离的项目更安全
    零门槛,包教会。让你在5分钟内使用以太坊ERC20智能合约发行属于自己的空气币
    spring boot高性能实现二维码扫码登录(下)——订阅与发布机制版
    spring boot高性能实现二维码扫码登录(中)——Redis版
    spring boot高性能实现二维码扫码登录(上)——单服务器版
  • 原文地址:https://www.cnblogs.com/wendelhuang/p/6795035.html
Copyright © 2020-2023  润新知