深入探索:单元测试
博客专区 > 灯-塔 的博客 > 博客详情
深入探索:单元测试
灯-塔 发表于6个月前
深入探索:单元测试
  • 发表于 6个月前
  • 阅读 8
  • 收藏 0
  • 点赞 0
  • 评论 0

新睿云服务器60天免费使用,快来体验!>>>   

摘要: 作者:灯塔

单元测试这一块我们一直都有在接触,但是对于我来说,还不够重视或深入,可以说是忽视的一块,现在公司要把测试覆盖率给推起来,此时正好可以来探究探究。 JUnit4 单元测试主要有以下的一些功能:

  • 参数测试
  • 异常测试
  • 超时测试
  • 灵活固件
  • 忽略测试
  • 对测试进行逻辑分组

注意:本版本基于JUnit4.12。

参数测试

下面我们会对这样的一个类做参数测试,我们需呀一个 参数 和 一个预期结果值

public class Fibonacci {
    public static int compute(int n) {
        int result = 0;

        if (n <= 1) {
            result = n;
        } else {
            result = compute(n - 1) + compute(n - 2);
        }

        return result;
    }
}

如何做测试?第一个方法是构造函数方式

@RunWith(Parameterized.class)
public class FibonacciTest {

    @Parameters
    public static Collection<Object[]> data() {
        /* 数组第一个 为 参数, 第二个为 期望结果 */
        return Arrays.asList(new Object[][] {{0, 0}, {1, 1}, {2, 1}, {3, 2}, {4, 3}, {5, 5}, {6, 8}});
    }

    private int fInput;

    private int fExpected;

    public FibonacciTest(int input, int expected) {
        fInput = input;
        fExpected = expected;
    }

    @Test
    public void test() {
        Assert.assertEquals(fExpected, Fibonacci.compute(fInput));
    }
}

非构造函数实现(推荐:非构造函数更加灵活,不会影响类的其他方法):

@RunWith(Parameterized.class)
public class FibonacciTest2 {
    @Parameters
    public static Collection<Object[]> data() {
        return Arrays.asList(new Object[][] {{0, 0}, {1, 1}, {2, 1}, {3, 2}, {4, 3}, {5, 5}, {6, 8}});
    }

    @Parameter // first data value (0) is default
    public /* NOT private */ int fInput; // 数组第一个参数为输入参数

    @Parameter(1)
    public /* NOT private */ int fExpected; // 数组第一个参数为 预期参数

    @Test
    public void test() {
        Assert.assertEquals(fExpected, Fibonacci.compute(fInput));
    }
}

参考:https://github.com/junit-team/junit4/wiki/Parameterized-tests

异常测试

在我们编写测试用例的过程中,我们往往都会有期望异常的测试方法,有异常就是对了。 举一个例子

new ArrayList<Object>().get(0);

这样的方法毫无疑问会爆出IndexOutOfBoundsException 数组越界异常,为此,我们要验证这个异常的出现是正确的,我们可以@Test(exceted=IndexOutOfBoundsException.class) 来标记期望的异常。

@Test(expected = IndexOutOfBoundsException.class) 
public void empty() { 
     new ArrayList<Object>().get(0); 
}

然而这个只是方法之一,但却不是推荐的方法,对于更长的测试,建议使用ExpectedException Rule 来实现;

对异常进行更深的测试: 上面的方式是一个非常好的示例,但是他是有限制的。例如, 不能在异常中测试消息的值, 也不能在引发异常后检测域对象的状态。在Junit3.x 是用 try/catch 来做的(这个不浪费时间叙述),但是在Junit4.x 可以使用 @Rule 来实现。使用 ExpectedException 规则。此规则允许您不仅指示所期望的异常, 而且还可以显示所期望的异常消息。 实现如下:

public class TestExyRule {
    @Rule
    public ExpectedException thrown = ExpectedException.none();

    @Test
    public void shouldTestExceptionMessage() throws IndexOutOfBoundsException {
        List<Object> list = new ArrayList<Object>();
        thrown.expect(IndexOutOfBoundsException.class);
        thrown.expectMessage("Index: 0, Size: 0");
        list.get(0); // execution will never get past this line
    }
}

上面例子的expectMessage 可以使用 Matchers 来做, 这样可以更加的灵活,如:thrown.expectMessage(Matchers.containsString("Size: 0")); 当然我们也可以更加进一步的 验证异常的状态:

import static org.hamcrest.Matchers.hasProperty;
import static org.hamcrest.Matchers.is;
import static org.hamcrest.Matchers.startsWith;

import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;

public class TestExy {
     @Rule
    public ExpectedException thrown = ExpectedException.none();

    @Test
    public void shouldThrow() {
        TestThing testThing = new TestThing();
        thrown.expect(UnsupportedOperationException.class);
        thrown.expectMessage(startsWith("some Message"));
        testThing.chuck();
    }

    private class TestThing {
        public void chuck() {
            throw new UnsupportedOperationException("some Message");
        }
    }
}

https://github.com/junit-team/junit4/wiki/Exception-testing

超时测试

https://github.com/junit-team/junit4/wiki/Timeout-for-tests 超时测试比较简单,举个例子。

@Test(timeout =1000)
public void testMethod() {
    for(;;){}
}

如上面的代码我们对时间做了限制为1s, 但是由于有死循环,是无法在规定时间内通过测试的。

灵活固件

https://github.com/junit-team/junit4/wiki/Aggregating-tests-in-suites

逻辑分组测试

https://github.com/junit-team/junit4/wiki/Categories

参考

标签: JUnit4 单元测试
  • 打赏
  • 点赞
  • 收藏
  • 分享
共有 人打赏支持
粉丝 2
博文 33
码字总数 53236
×
灯-塔
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
* 金额(元)
¥1 ¥5 ¥10 ¥20 其他金额
打赏人
留言
* 支付类型
微信扫码支付
打赏金额:
已支付成功
打赏金额: