`
namavaj
  • 浏览: 8461 次
  • 性别: Icon_minigender_1
  • 来自: 北京
最近访客 更多访客>>
社区版块
存档分类
最新评论

[转载]详细讲解在Spring中进行集成测试七(通过JDBC访问数据库,检测数据操作正确性)

阅读更多
让测试方法自动拥有回滚能力
    AbstractTransactionalSpringContextTests 专为解决以上问题而生,也就是说前面我们所提及的第3)个问题在此得到了回答。只要继承该类创建测试用例,在默认情况下,测试方法中所包含的事务性数据操 作都会在测试方法返回前被回滚。由于事务回滚操作发生在测试方法返回前的点上,所以你可以象往常一样在测试方法体中对数据操作的正确性进行校验。
java 代码
 
  1. package com.baobaotao.service;  
  2. import org.springframework.test.AbstractTransactionalSpringContextTests;   
  3. import com.baobaotao.domain.User;   
  4. public class UserServiceIntegrateTest extends AbstractTransactionalSpringContextTests {   
  5.     private UserService userService;   
  6.     public void setUserService(UserService userService) {   
  7.         this.userService = userService;  
  8.      }   
  9.     @Override   
  10.     protected String[] getConfigLocations() {   
  11.         return new String[]{"baobaotao-service.xml""baobaotao-dao.xml"};   
  12.     }   
  13.     public void testRegisterUser(){ ①测试方法中的数据操作将在方法返回前被回滚,不会对数据库      
  14.         User user = new User(); 产生永久性数据操作,第二次运行该测试方法时,依旧可以    
  15.         user.setUserId(2); 成功运行。   
  16.         user.setUserName("john");   
  17.         user.setPassword("123456");   
  18.         userService.registerUser(user);   
  19.         User user1 = userService.findUserByUserName("john"); ②对数据操作进行     
  20.         assertEquals(user.getUserId(), user1.getUserId()); 正确性检验   
  21.     }   
  22. }   
    如果testRegisterUser()是直接继承于AbstractDependencyInjectionSpringContextTests类 的测试方法,则重复运行该测试方法就会发生数据冲突问题。但因为它位于继承于 AbstractTransactionalSpringContextTests的测试用例类中,测试方法中对数据库的操作会被正确回滚,所以重复运行 不会有任何问题。
    如果你确实希望测试方法中对数据库的操作持久生效而不是被回滚,Spring也可以满足你的要求,你仅需要在测试方法中添加setComplete()方法就可以了。
   
java 代码
 
  1. public void testRegisterUser(){  
  2. …  
  3.     User user1 = userService.findUserByUserName("john");  
  4.     assertEquals(user.getUserId(), user1.getUserId());  
  5.     setComplete(); ①测试方法中的事务性数据操作将被提交  
  6. }   

    AbstractTransactionalSpringContextTests还拥有几个可用于初始化测试数据库,并在测试完成后清除测试数据的方法,分别介绍如下:
     onSetUpBeforeTransaction()/onTearDownAfterTransaction():子类可以覆盖这两个方法,以便在事务性测试方法运行的前后执行一些数据库初始化的操作并在事务完成后清除之;
     onSetUpInTransaction()/onTearDownInTransaction():这对方法和前面介绍的方法完成相同的功能,只不过它们是在测试方法的相同事务中执行的。
    AbstractTransactionalSpringContextTests 另外还提供了一组用于测试延迟数据加载的方法:endTransaction()/startNewTransaction()。我在测试 Hibernate、JPA等允许延迟数据加载的应用时,如何模拟数据在Service层事务中被部分加载,当传递到Web层时重新打开事务完成延迟部分 数据加载的测试场景呢?这两个方法即为此用途而生:你可以在测试方法中显式调用endTransaction()方法以模拟从Service层中获取部分 数据后返回,尔后,再通过startNewTransaction()开启一个和原事务无关新事务——模拟在Web层中重新打开事务,接下来你就可以访问 延迟加载的数据,看是否一切如期所料了。
    在代码清单 6的②处,我们通过UserService#findUserByUserName()方法对前面registerUser(user)方法数据操作的正 确性进行检验。应该说,我们非常幸运,因为在UserService中刚好存在一个可用于检测registerUser(user)数据操作正确性的方 法。让我们考虑另外的一种情况:要是 UserService不存在这样的方法,我们该如何检测registerUser(user)数据操作结果的正确性呢?显然我们不能使用肉眼观察的方 法,那难道为了验证数据操作正确性专门编写一个配合性的数据访问类不成?

通过JDBC访问数据库,检测数据操作正确性
    正当我们“山重水复疑无路”的时候,让我们再往前走上一程,柳暗花明将倏忽而至
—— AbstractTransactionalDataSourceSpringContextTests就是花开景明之所。该类继承于 AbstractTransactionalSpringContextTests,它添加了一个JdbcTemplate,你可以借由此道快意直达数据 库。它自动使用Spring容器中的数据源(DataSource)创建好一个JdbcTemplate实例并开放给子类使用。值得注意的是,如果你采用 byName自动装配机制,数据源Bean的名称必须取名为“dataSource”。
    让我们对UserServiceIntegrateTest进行改造,以便让其自动拥有访问数据库的设施(JdbcTemplate),并用灵活的方法访问数据库进行数据操作的检验,其代码如下所示:
java 代码
 
  1. package com.baobaotao.service;  
  2. import org.springframework.test.AbstractTransactionalDataSourceSpringContextTests;   
  3. …   
  4. public class UserServiceIntegrateWithJdbcTest extends AbstractTransactionalDataSourceSpringContextTests {①注意:继承类发生调整   
  5.     private UserService userService;  
  6.     public void setUserService(UserService userService) {   
  7.         this.userService = userService;   
  8.     }   
  9.     @Override   
  10.     protected String[] getConfigLocations() {   
  11.         return new String[]{"baobaotao-service.xml""baobaotao-dao.xml"};   
  12.     }   
  13.     public void testRegisterUser(){   
  14.         User user = new User(); user.setUserId(2);  
  15.         user.setUserName("john");   
  16.         user.setPassword("123456");  
  17.         userService.registerUser(user);   
  18.         String sqlStr = " SELECT user_id FROM t_user WHERE user_name ='john' ";  
  19.         int userId = jdbcTemplate.queryForInt(sqlStr); ①可以直接使用JdbcTemplate访问数据库了       
  20.         assertEquals(user.getUserId(), userId); setComplete();   
  21.     }  
  22. }  
    jdbcTemplate是AbstractTransactionalDataSourceSpringContextTests类中定义的,子类可以 直接使用它访问数据库。这样我们就可以灵活地访问数据库以检验目标测试方法的数据操作正确性。至此,我们终于毕其功于一役于 AbstractTransactionalDataSourceSpringContextTests,顺利解决前面我们中指出的最后问题。
    只 要你通过扩展AbstractTransactionalSpringContextTests及其子类创建测试用例,所有测试方法都会工作了事务环境 下。也就是说,即使某些测试方法不需要访问数据库,也会产生额外的事务管理开销,是否可以对测试方法启用事务管理的行为进行控制呢?此外,在一些情况下, 除对目标方法逻辑运行的正确性进行检验外,我们还希望对目标方法的运行性能进行测试:如当目标方法运行时间超过200毫秒时,则测试用例视为未通过。诸如 此类的问题,我们目前学习到的知识还不能很好的应付。Spring 2.0新增了注解驱动的测试工具为我们指明了道路,你仅需要通过简单为测试方法标注注解,我们刚才提出的“疑难”问题就可以迎刃而解了。
小结
    本文我们讲述了使用Spring提供的一套测试工具对Spring应用程序进行集成测试所需的所有知识。
Spring 建议你不应该在单元测试时使用到Spring容器,你应该在集成测试时才使用到Spring容器。手工创建测试固件或者手工装配测试固件的工作都是单调乏 味没有创意的工作,通过使用Spring为集成测试提供了帮助类,你就可以享受测试固件自动装配的好处,将精力集中到目标类逻辑测试编写的工作上。
    应 该说大部分的Java应用都是Web应用,而大部分的Java Web应用都是数据库相关的应用,对数据库应用进行测试经常要考虑数据准备、数据库现场恢复、灵活访问数据以验证数据操作正确性等等的问题。这些问题如果 没有一个很好的支持工具,将给编写测试用例造成挑战,幸好Spring都为我们搭建好满足这些需求的测试平台,你仅需要在此基础上编写特定的测试用例就可 以了。
分享到:
评论

相关推荐

    支持多数据库的ORM框架ef-orm.zip

    从数据库元数据访问,到建表,创建约束,创建sequence等各种DDL操作进行了封装,用户无需编写各种SQL,可以直接通过API操作数据库结构。 尤其是ALTER TABLE等修改数据库的语句,各种不同的RDBMS都有较大语法差异。...

    本科毕业设计开题报告(食得客快餐店订餐系统的设计与实现)

    在软件开发过程中,解决技术问题使用的方法是文献法,通过查阅课本、图书馆资料和网络在线文献、网络在线案例参考等,解决在软件开发过程中的技术问题,比如数据库、建模工具的使用、环境搭建、软件测试等。...

    Activiti6.0教程例子下载

    4. TaskService: 在Activiti中业务流程定义中的每一个执行节点被称为一个Task,对流程中的数据存取,状态变更等操作均需要在Task中完成。TaskService提供了对用户Task 和Form相关的操作。它提供了运行时任务查询、...

    JAVA上百实例源码以及开源项目源代码

    FTP的目标是:(1)提高文件的共享性(计算机程序和/或数据),(2)鼓励间接地(通过程序)使用远程计算机,(3)保护用户因主机之间的文件存储系统导致的变化,(4)为了可靠和高效地传输,虽然用户可以在终端上...

    java开源包1

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    java开源包11

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    java开源包2

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    java开源包3

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    java开源包6

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    java开源包5

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    java开源包10

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    java开源包4

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    java开源包8

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    java开源包7

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    java开源包9

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    JAVA上百实例源码以及开源项目

    FTP的目标是:(1)提高文件的共享性(计算机程序和/或数据),(2)鼓励间接地(通过程序)使用远程计算机,(3)保护用户因主机之间的文件存储系统导致的变化,(4)为了可靠和高效地传输,虽然用户可以在终端上...

    java开源包101

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

    Java资源包01

    利用JoSQL可以像操作数据库中的数据一样对任何Java对象集进行查询,排序,分组。 搜索自动提示 Autotips AutoTips是为解决应用系统对于【自动提示】的需要(如:Google搜索), 而开发的架构无关的公共控件, 以满足该类...

Global site tag (gtag.js) - Google Analytics