本着“不写单元测试的程序员不是好程序员”原则,我在坚持写着单元测试,不敢说全部的Java web应用都基于Spring,但至少一半以上都是基于Spring的。
发现经过Spring进行bean管理后,作测试会有各类不足,
例如,不少人作单元测试的时候,还要在Before方法中,初始化Spring容器,致使容器被初始化屡次。
- @Before
- public void init() {
- ApplicationContext ctx = new FileSystemXmlApplicationContext( "classpath:spring/spring-basic.xml");
- baseDao = (IBaseDao) ctx.getBean("baseDao");
- assertNotNull(baseDao);
- }
在开发基于Spring的应用时,若是你还直接使用Junit进行单元测试,那你就错过了Spring满汉全席中最重要的一道硬菜。html
再说这道菜以前,咱们先来讨论下,在基于Spring的javaweb项目中使用Junit直接进行单元测试有什么不足
1)致使屡次Spring容器初始化问题
根据JUnit测试方法的调用流程,每执行一个测试方法都会建立一个测试用例的实例并调用setUp()方法。因为通常状况下,咱们在setUp()方法中初始化Spring容器,这意味着若是测试用例有多少个测试方法,Spring容器就会被重复初始化屡次。虽然初始化Spring容器的速度并不会太慢,但因为可能会在Spring容器初始化时执行加载Hibernate映射文件等耗时的操做,若是每执行一个测试方法都必须重复初始化Spring容器,则对测试性能的影响是不容忽视的;
/////////使用Spring测试套件,Spring容器只会初始化一次!
2)须要使用硬编码方式手工获取Bean
在测试用例类中咱们须要经过ctx.getBean()方法从Spirng容器中获取须要测试的目标Bean,而且还要进行强制类型转换的造型操做。这种乏味的操做迷漫在测试用例的代码中,让人以为烦琐不堪;
////////使用Spring测试套件,测试用例类中的属性会被自动填充Spring容器的对应Bean
,无须在手工设置Bean!
3)数据库现场容易遭受破坏
测试方法对数据库的更改操做会持久化到数据库中。虽然是针对开发数据库进行操做,但若是数据操做的影响是持久的,可能会影响到后面的测试行为。举个例子,用户在测试方法中插入一条ID为1的User记录,第一次运行不会有问题,第二次运行时,就会由于主键冲突而致使测试用例失败。因此应该既可以完成功能逻辑检查,又可以在测试完成后恢复现场,不会留下“后遗症”;
////////使用Spring测试套件,Spring会在你验证后,自动回滚对数据库的操做,保证数据库的现场不被破坏,所以重复测试不会发生问题!
4)不方便对数据操做正确性进行检查
假如咱们向登陆日志表插入了一条成功登陆日志,但是咱们却没有对t_login_log表中是否确实添加了一条记录进行检查。通常状况下,咱们多是打开数据库,肉眼观察是否插入了相应的记录,但这严重违背了自动测试的原则。试想在测试包括成千上万个数据操做行为的程序时,如何用肉眼进行检查?
////////只要你继承Spring的测试套件的用例类,你就能够经过jdbcTemplate在同一事务中访问数据库,查询数据的变化,验证操做的正确性!
看完上面的内容,相信,你已经知道我说的硬菜是什么了。
下面,让咱们看看,使用Spring测试套件后,代码是如何变优雅的。
1. 加入依赖包
使用spring的测试框架须要加入如下依赖包:java
- JUnit 4
- 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>
2. 建立测试源目录和包
在此,推荐建立一个和src平级的源文件目录,由于src内的类都是为往后产品准备的,而此处的类仅仅用于测试。而包的名称能够和src中的目录同名,这样因为在test源目录中,因此不会有冲突,并且名称又如出一辙,更方便检索。这也是Maven的约定。程序员
三、建立测试类
1)基类,其实就是用来加载配置文件的
- @RunWith(SpringJUnit4ClassRunner.class)
- @ContextConfiguration
- ({"/spring/app*.xml","/spring/service/app*.xml"})
-
- public class BaseJunit4Test {
- }
2)接着是咱们本身的测试类
- public class UserAssignServiceTest extends BaseJunit4Test{
- @Resource
- private IBaseDao baseDao;
- @Test
- @Transactional
- @Rollback(false)
- public void insert( ) {
- String sql="insert into user(name,password) values(?,?)";
- Object[] objs=new Object[]{"00","000"};
- baseDao.insert( sql , objs );
- String sql1="select * from user where name=? and password=? ";
- List<Map<String,Object>> list=baseDao.queryForList( sql1 , objs );
- System.out.println(list);
- assertTrue(list.size( )>0);
- }
- }
原文:http://blog.csdn.net/shan9liang/article/details/40452469web