Junit初级篇
@Test
介绍
@Test
是我们在写测试脚本时最常用到的,大部分情况下如果没用这个注解,一个方法就不能成为测试用例。如下代码是一个最普通的测试脚本:
import org.junit.Assert;
import org.junit.Test; public class GeneralTest { @Test
public void test() {
int num1 = 1;
int num2 = 2;
int sum = num1 + num2;
Assert.assertEquals(3, sum);
} }
运行一下看看:
恭喜,获得一个绿条。这个脚本包含了测试脚本基本的几个元素:参数定义、参数运算、结果校验。@Test
在这个测试脚本的作用就是让一个普通的方法(Method)变成一个JUnit可识别的测试脚本。但是@Test
除了标记测试脚本的功能外,还有什么功能呢?
先来看看@Test
的源码:
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; @Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface Test { /**
* Default empty exception
*/
static class None extends Throwable {
private static final long serialVersionUID = 1L; private None() {
}
} /**
* Optionally specify <code>expected</code>, a Throwable,
* to cause a test method to succeed iff
* an exception of the specified class is thrown by the method.
*/
Class<? extends Throwable> expected() default None.class; /**
* Optionally specify <code>timeout</code> in milliseconds to
* cause a test method to fail if it
* takes longer than that number of milliseconds.
*/
long timeout() default 0L;
}
从源码可以看出,@Test
还有两个属性,分别是expected()
和timeout()
,可以从注解中看到这两个属性的作用:
expected()
:校验测试脚本抛出指定类型的异常timeout()
:测试脚本的运行时间是否满足要求
下面举个例子看看这个两个属性如何使用:
import org.junit.Test; public class TimeoutAndExceptionTest { @Test(timeout = 1000) //脚本运行时间小于1000毫秒
public void testTimeout() throws InterruptedException {
Thread.sleep(900);
} @Test(expected = IllegalArgumentException.class) //脚本抛出参数非法的异常
public void testExpectedThrowable() {
throw new IllegalArgumentException();
} @Test(timeout = 1000) //脚本运行时间小于1000毫秒
public void testTimeoutFailed() throws InterruptedException {
Thread.sleep(2000);
} @Test(expected = IllegalArgumentException.class) //脚本抛出参数非法的异常
public void testExpectedThrowableFailed() {
throw new NullPointerException();
} }
运行后发现testTimeoutFailed()
和testExpectedThrowableFailed()
运行失败,错误提示如下:
可见JUnit已经帮助我们完成断言的工作。至此,@Test
的功能已经全部介绍完了。
@Ignore
介绍
@Ignore
是偶尔且一定会用到的注解,从名字就能知道其作用是“忽略”,即忽略一个测试脚本,标注了@Ignore
注解的测试脚本将不会被执行。用法很简单,示例如下:
import org.junit.Ignore;
import org.junit.Test; public class IgnoreTest { @Test
@Ignore
public void test() {
//some test
} }
运行一下
发现这个测试脚本的确没被运行。不过如果另外一个人来看这个脚本就会很疑惑这个测试脚本为什么要被忽略。所以必须要说明这个测试脚本被忽略的原因。那在什么地方说明呢?可以先看看@Ignore
本身是否提供这个功能,所以先看看@Ignore
的源码:
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; @Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD, ElementType.TYPE})
public @interface Ignore {
/**
* The optional reason why the test is ignored.
*/
String value() default "";
}
从源码来看,@Ignore
的value()
就是为这个需求准备的,可以说明测试脚本被忽略的原因。我们可以将上面的测试脚本修改如下:
import org.junit.Ignore;
import org.junit.Test; public class IgnoreTest { @Test
@Ignore("not ready yet!")
public void test() {
//some test
} }
修改后就清楚多了,无论以后谁来维护这个测试脚本,都可以很直观的看到这个测试脚本是因为“还未准备好”而忽略的。
再运行一下
发现运行结构并没有不同,可见Eclipse并有没读取@Ignore
的value()
值,不过并不是所有的IDE都是这样,可以看下IDEA的运行效果。
从运行结果发现,IDEA可以在运行结果展示被忽略的原因。
@Before
、@After
、@BeforeClass
、@AfterClass
介绍
@Before
、@After
、@BeforeClass
、@AfterClass
也是我们常用的注解,使用频率与@Test
一样,一般每个测试脚本都会出现这些注解,用来做测试数据准备和资源回收等工作。那么,首先介绍下这些注解的执行范围。
@Bfore
:每个测试用例(Method)运行前执行一次@After
:每个测试用例(Method)运行后执行一次@BeforeClass
:每个测试脚本(Class)运行前执行一次@AfterClass
:每个测试脚本(Class)运行前执行一次
每个注解对所标注的方法(Method)的要求如下:
@Bfore
、@After
:公共(public)的、非静态(static)的、无返回值(void)、无参数的方法(Method)@BeforeClass
、@AfterClass
:公共(public)的、静态(static)的、无返回值(void)、无参数的方法(Method)
下面用一个简单的例子说明一下这些规则:
import org.junit.After;
import org.junit.AfterClass;
import org.junit.Before;
import org.junit.BeforeClass;
import org.junit.Test; public class BeforeAndAfterTest { @BeforeClass
public static void beforeClass() {
System.out.println("BeforeClass");
} @AfterClass
public static void afterClass() {
System.out.println("AfterClass");
} @Before
public void before() {
System.out.println("Before");
} @After
public void after() {
System.out.println("After");
} @Test
public void test01() {
System.out.println("Test01");
} @Test
public void test02() {
System.out.println("Test02");
} }
运行一下
只有标注了@Test
注解的方法(Method)被识别为测试用例。
再看看控制台(Console)的输出内容
BeforeClass
Before
Test01
After
Before
Test02
After
AfterClass
和上面所说的规则一致。
@RunWith
介绍
基于JUnit的测试脚本都是依靠Runner去解析和执行,@Test
等所有注解的执行顺序、生命周期也都是由Runner决定的,所以Runner是JUnit的核心。
自然,将Runner作为JUnit的扩展点将使JUnit更易扩展,这个将在高级篇介绍。
从JUnit4.5以后,JUnit的默认Runner改为BlockJUnit4ClassRunner
,将原来的JUnit4ClassRunner
改为BlockJUnit4ClassRunner
继承ParentRunner<T>
的形式,增强了JUnit的可扩展性。同时,JUnit也自带很多Runner来丰富自身的功能。
那么,使用其他功能的Runner呢?那就要使用@RunWith
注解了,下面用一个简单的小例子演示下@RunWith
的使用。
首先需要自定义一个Runner,作用是打印每个运行的测试用例的方法名(Method Name),具体Runner的实现将在高级篇中详细讨论。代码如下:
import org.junit.runner.notification.RunNotifier;
import org.junit.runners.BlockJUnit4ClassRunner;
import org.junit.runners.model.FrameworkMethod;
import org.junit.runners.model.InitializationError; public class LoggingRunner extends BlockJUnit4ClassRunner { public LoggingRunner(Class<?> klass) throws InitializationError {
super(klass);
} @Override
protected void runChild(FrameworkMethod method, RunNotifier notifier) {
System.out.println(method.getName());
super.runChild(method, notifier);
} }
然后通过@RunWith
使用这个注解。
import org.junit.Test;
import org.junit.runner.RunWith; @RunWith(LoggingRunner.class) //在这里使用
public class RunWithTest { @Test
public void test01() {
} @Test
public void test02() {
} }
最后运行下看看结果。
test01
test02
正确打印出两个测试用例的方法。
可以看到,@RunWith
用法十分简单,但是这是个十分重要的功能,因为后面的中级篇、高级篇中大部分功能都基于一个特殊功能的Runner,都需要使用@RunWith
,请牢记。
摘自:http://liwx2000.github.io/junit/2013/05/15/junit-junior.html
Junit初级篇的更多相关文章
- Java工程师学习指南 初级篇
Java工程师学习指南 初级篇 最近有很多小伙伴来问我,Java小白如何入门,如何安排学习路线,每一步应该怎么走比较好.原本我以为之前的几篇文章已经可以解决大家的问题了,其实不然,因为我之前写的文章都 ...
- Java工程师学习指南(初级篇)
Java工程师学习指南 初级篇 最近有很多小伙伴来问我,Java小白如何入门,如何安排学习路线,每一步应该怎么走比较好.原本我以为之前的几篇文章已经可以解决大家的问题了,其实不然,因为我之前写的文章都 ...
- Python 正则表达式入门(初级篇)
Python 正则表达式入门(初级篇) 本文主要为没有使用正则表达式经验的新手入门所写. 转载请写明出处 引子 首先说 正则表达式是什么? 正则表达式,又称正规表示式.正规表示法.正规表达式.规则表达 ...
- python 面向对象初级篇
Python 面向对象(初级篇) 概述 面向过程:根据业务逻辑从上到下写垒代码 函数式:将某功能代码封装到函数中,日后便无需重复编写,仅调用函数即可 面向对象:对函数进行分类和封装,让开发" ...
- 25个增强iOS应用程序性能的提示和技巧(初级篇)
25个增强iOS应用程序性能的提示和技巧(初级篇) 标签: ios内存管理性能优化 2013-12-13 10:53 916人阅读 评论(0) 收藏 举报 分类: IPhone开发高级系列(34) ...
- ASP.NET MVC 随想录——开始使用ASP.NET Identity,初级篇(转)
ASP.NET MVC 随想录——开始使用ASP.NET Identity,初级篇 阅读目录 ASP.NET Identity 前世今生 建立 ASP.NET Identity 使用ASP.NET ...
- python_way ,day7 面向对象 (初级篇)
面向对象 初级篇 python支持 函数 与 面向对象 什么时候实用面向对象? 面向对象与函数对比 类和对象 创建类 class 类名 def 方法名(self,xxxx) 类里面的方法,只能 ...
- Entity Framework 学习初级篇--基本操作:增加、更新、删除、事务(转)
摘自:http://www.cnblogs.com/xray2005/archive/2009/05/17/1458568.html 本节,直接写通过代码来学习.这些基本操作都比较简单,与这些基本操作 ...
- NSIS安装制作基础教程[初级篇], 献给对NSIS有兴趣的初学者
NSIS安装制作基础教程[初级篇], 献给对NSIS有兴趣的初学者 作者: raindy 来源:http://bbs.hanzify.org/index.php?showtopic=30029 时间: ...
随机推荐
- LeetCode741. Cherry Pickup
https://leetcode.com/problems/cherry-pickup/description/ In a N x N grid representing a field of che ...
- MySQL-事务特性
1. 事务概念引入: 现实生活中,我们往往经常会进行转账操作,转账操作可以分为两部分来完成,转入和转出.只有这两部分都完成了才可以认为是转账成功.在数据库中,这个过程是使用两条语句来完成的,如果其中任 ...
- 洛谷 P2369 EXCEEDED WARNING A 题解
题目传送门 直接用sort排序最后输出即可.但是数组要使用short int 类型.否则会超内存. #include<bits/stdc++.h> using namespace std; ...
- Fresco 获得Bitmap
ImageRequest imageRequest = ImageRequestBuilder .newBuilderWithSource( Uri.parse(getFeedItem(positio ...
- linux中top命令
经常用到top命令,也就简单看看进程多不多,卡不卡, 这次在网上找到一个归纳总结的,以供参考. 简介 top 命令是 Linux 下常用的性能分析工具,能够实时显示系统中各个进程的资源占用状况,类似于 ...
- SpringBoot入门系列
集合redis,mysql,测试例子 http://blog.csdn.net/lxhjh/article/details/51764604
- 转:win32下的堆管理系统
转:https://bbs.pediy.com/thread-224136.htm 准备刷漏洞战争中的堆溢出部分,但是对于堆的了解较少,在这里记录一下关于堆的学习记录,有错误请各位大大拍砖 参考: & ...
- [hdu3685]Rotational Painting 凸包 重心
大致题意: 给出一个多边形,问你有多少种放法可以使得多边形稳定得立在平面上. 先对多边形求重心,在求凸包,枚举凸包的边,如果重心没有在边的范围内,则不行 判断是否在范围内可用点积来判断 #includ ...
- 洛谷P1073 最优贸易 [图论,DP]
题目传送门 最优贸易 题目描述 C 国有n 个大城市和m 条道路,每条道路连接这n 个城市中的某两个城市.任意两个城市之间最多只有一条道路直接相连.这m 条道路中有一部分为单向通行的道路,一部分为双向 ...
- java.io 文件分类
文件分为二进制格式和文本格式. 数据在计算机中都是以二进制的形式表现的,一般来说字节是最小的数据逻辑单位,所以也可以说数据都是以字节序列的形式表现的,不管是在内存中还是磁盘文件中.如果直接把内存中的数 ...