我们在编写大型程序的时候,需要写成千上万个方法或函数,这些函数的功能可能很强大,但我们在程序中只用到该函数的一小部分功能,并且经过调试可以确定,这一小部分功能是正确的。但是,我们同时应该确保每一个函数都完全正确,因为如果我们今后如果对程序进行扩展,用到了某个函数的其他功能,而这个功能有bug的话,那绝对是一件非常郁闷的事情。所以说,每编写完一个函数之后,都应该对这个函数的方方面面进行测试,这样的测试我们称之为单元测试。传统的编程方式,进行单元测试是一件很麻烦的事情,你要重新写另外一个程序,在该程序中调用你需要测试的方法,并且仔细观察运行结果,看看是否有错。正因为如此麻烦,所以程序员们编写单元测试的热情不是很高。于是有一个牛人推出了单元测试包,大大简化了进行单元测试所要做的工作,这就是JUnit4。
在Eclipse3.2中使用JUnit4进行单元测试
首先,我们来一个傻瓜式速成教程,不要问为什么,Follow Me,先来体验一下单元测试的快感!
首先新建一个项目叫JUnit_Test,我们编写一个Calculator类,这是一个能够简单实现加减乘除、平方、开方的计算器类,然后对这些功能进行单元测试。这个类并不是很完美,我们故意保留了一些Bug用于演示,这些Bug在注释中都有说明。该类代码如下:
public class Calculator {
private static int result; // 静态变量,用于存储运行结果
public void add( int n) {
result = result + n;
}
public void substract( int n) {
result = result - 1 ; //Bug: 正确的应该是 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 (; ;) ; //Bug : 死循环
}
public void clear() { // 将结果清零
result = 0 ;
}
public int getResult() {
return result;
}
}
第二步,将JUnit4单元测试包引入这个项目:在该项目上点右键,点“属性”,如图:
在弹出的属性窗口中,首先在左边选择“ Java Build Path ”,然后到右上选择“ Libraries ”标签,之后在最右边点击“ Add Library… ”按钮,如下图所示:
然后在新弹出的对话框中选择 JUnit4 并点击确定,如上图所示, JUnit4 软件包就被包含进我们这个项目了。
第三步,生成 JUnit 测试框架:在 Eclipse 的 Package Explorer 中用右键点击该类弹出菜单,选择“ New à JUnit Test Case ”。如下图所示:
在弹出的对话框中,进行相应的选择,如下图所示:
点击“下一步”后,系统会自动列出你这个类中包含的方法,选择你要进行测试的方法。此例中,我们仅对“加、减、乘、除”四个方法进行测试。如下图所示:
之后系统会自动生成一个新类CalculatorTest,里面包含一些空的测试用例。你只需要将这些测试用例稍作修改即可使用。完整的CalculatorTest代码如下:
import static org.junit.Assert. * ;
import org.junit.Before;
import org.junit.Ignore;
import org.junit.Test;
public class CalculatorTest {
private static Calculator calculator = new Calculator();
@Before
public void setUp() throws Exception {
calculator.clear();
}
@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());
}
@Ignore( " Multiply() Not yet implemented " )
@Test
public void testMultiply() {
}
@Test
public void testDivide() {
calculator.add( 8 );
calculator.divide( 2 );
assertEquals( 4 , calculator.getResult());
}
}
第四步,运行测试代码:按照上述代码修改完毕后,我们在CalculatorTest类上点右键,选择“Run As àJUnit Test”来运行我们的测试,如下图所示:
运行结果如下:
进度条是红颜色表示发现错误,具体的测试结果在进度条上面有表示“共进行了4个测试,其中1个测试被忽略,一个测试失败”。
Spring整合Junit4进行单元测试
一.加入依赖包
使用spring的测试框架需要加入以下依赖包:
JUnit 4 (官方下载:http://www.junit.org/)
Spring Test (Spring框架中的test包)
Spring 相关其他依赖包(不再赘述了,就是context等包)
如果使用maven,在基于spring的项目中添加如下依赖:
- <dependency>
- <groupId>junit</groupId>
- <artifactId>junit</artifactId>
- <version>4.9</version>
- <scope>test</scope>
- </dependency>
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-test</artifactId>
- <version> 3.2.4.RELEASE </version>
- <scope>provided</scope>
- </dependency>
二.创建测试类
1)基类,其实就是用来加载配置文件的
- package Solin.Test;
- import org.junit.runner.RunWith;
- import org.springframework.test.context.ContextConfiguration;
- import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
- @RunWith(SpringJUnit4ClassRunner.class) //使用junit4进行测试
- @ContextConfiguration(locations={"classpath:applicationContext.xml"}) //加载配置文件
- //------------如果加入以下代码,所有继承该类的测试类都会遵循该配置,也可以不加,在测试类的方法上///控制事务,参见下一个实例
- //这个非常关键,如果不加入这个注解配置,事务控制就会完全失效!
- //@Transactional
- //这里的事务关联到配置文件中的事务控制器(transactionManager = "transactionManager"),同时//指定自动回滚(defaultRollback = true)。这样做操作的数据才不会污染数据库!
- //@TransactionConfiguration(transactionManager = "transactionManager", defaultRollback = true)
- //------------
- public class BaseJunit4Test{
- }
解释下用到的注解:
@RunWith:用于指定junit运行环境,是junit提供给其他框架测试环境接口扩展,为了便于使用spring的依赖注入,spring提供了org.springframework.test.context.junit4.SpringJUnit4ClassRunner作为Junit测试环境
@ContextConfiguration({"classpath:applicationContext.xml","classpath:spring/buyer/applicationContext-service.xml"})
导入配置文件,这里我的applicationContext配置文件是根据模块来分类的。如果有多个模块就引入多个“applicationContext-service.xml”文件。如果所有的都是写在“applicationContext.xml”中则这样导入:
@ContextConfiguration(locations = "classpath:applicationContext.xml")
@TransactionConfiguration(transactionManager = "transactionManager", defaultRollback = true)这里的事务关联到配置文件中的事务控制器(transactionManager = "transactionManager"),同时指定自动回滚(defaultRollback = true)。这样做操作的数据才不会污染数据库!
@Transactional:这个非常关键,如果不加入这个注解配置,事务控制就会完全失效!
AbstractTransactionalDataSourceSpringContextTests要想构建这一系列的无污染纯绿色事务测试框架就必须找
到这个基类!(即所有事务均不生效)
2)接着是我们自己的测试类- package Solin.Test;
- import java.util.List;
- import org.junit.Test;
- import org.springframework.beans.factory.annotation.Autowired;
- import org.springframework.test.annotation.Rollback;
- import org.springframework.transaction.annotation.Transactional;
- import Solin.Entity.ImageInfo;
- import Solin.Service.ImageInfoService;
- public class ImageInfoTest extends BaseJunit4Test{
- @Autowired //自动注入
- private ImageInfoService imageInfoService;
- @Test
- @Transactional //标明此方法需使用事务
- @Rollback(false) //标明使用完此方法后事务不回滚,true时为回滚
- public void test(){
- System.out.println("测试Spring整合Junit4进行单元测试");
- ImageInfo imageInfo = new ImageInfo();
- imageInfo.setParentID(999);
- imageInfo.setImgAddr("地球");
- imageInfoService.saveImageInfo(imageInfo);
- List<ImageInfo> list = imageInfoService.getImageInfoList(95);
- for(ImageInfo img : list){
- System.out.println("parentID:"+img.getParentID()+"------imgAddr:"+img.getImgAddr());
- }
- }
- }
三.通过JUnit 4 执行
右键方法名,选择则“Run As”→“JUnit Test”即可
java1.8版本只支持spring4.0以上。所以避免运行异常的解决方法:①把sping版本换成4.0以上②把jdk调低点
此处我选择把jdk调低点,运行,结果如下:
- 测试Spring整合Junit4进行单元测试
- parentID:95------imgAddr:广东深圳
- parentID:95------imgAddr:广东深圳