一、DbUnit设计理念
熟悉单元测试的开发人员都知道,在对数据库进行单元测试时候,通常采用的方案有运用模拟对象(mock objects)和stubs两种。通过隔离关联的数据库访问类,比如JDBC的相关操作类,来达到对数据库操作的模拟测试。然而某些特殊的系统,比如利 用了EJB的CMP(container-managed persistence)的系统,数据库的访问对象是在最底层而且很隐蔽的,那么这两种解决方案对这些系统就显得力不从心了。
DBUnit的设计理念就是在测试之前,备份数据库,然后给对象数据库植入我们需要的准备数据,最后,在测试完毕后,读入备份数据库,回溯到测试前的状态;而且又因为DBUnit是对JUnit的一种扩展,开发人员可以通过创建测试用例代码,在这些测试用例的生命周期内来对数据库的操作结果进行比较。

二、DbUnit测试基本概念和流程
基于DbUnit 的测试的主要接口是IDataSet。IDataSet代表一个或多个表的数据。
可以将数据库模式的全部内容表示为单个IDataSet 实例。这些表本身由Itable 实例来表示。
IDataSet 的实现有很多,每一个都对应一个不同的数据源或加载机制。最常用的几种 IDataSet实现为:

FlatXmlDataSet:数据的简单平面文件 XML 表示
QueryDataSet:用 SQL 查询获得的数据
DatabaseDataSet:数据库表本身内容的一种表示
XlsDataSet :数据的excel表示

DBUnit支持的数据库包括,db2,h2,hsqldb,mckoi,mssql,mysql,netezza,oralce,postgresql.

三、DBUnit测试流程:

一般而言,使用DbUnit进行单元测试的流程如下:
1 根据业务,做好测试用的准备数据和预想结果数据,通常准备成xml格式文件。
2 在setUp()方法里边备份数据库中的关联表。
3 在setUp()方法里边读入准备数据。
4 对测试类的对应测试方法进行实装:执行对象方法,把数据库的实际执行结果和预想结果进行比较。
5 在tearDown()方法里边,把数据库还原到测试前状态。

 四、实例开发:

1、新建一个pom工程,加入相关依赖:

    <dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.11</version><scope>test</scope></dependency><dependency><groupId>org.dbunit</groupId><artifactId>dbunit</artifactId><version>2.5.3</version></dependency><dependency><groupId>org.slf4j</groupId><artifactId>slf4j-nop</artifactId><version>1.7.24</version></dependency><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><version>5.1.38</version></dependency>

2、新建工程相关目录

BaseDao.java文件:

public class BaseDao {private static Connection CONNECTION_INSTANCE = null;protected Connection getConnection() throws Exception{if(null==CONNECTION_INSTANCE){Class.forName("com.mysql.jdbc.Driver");CONNECTION_INSTANCE=DriverManager.getConnection("jdbc:mysql://127.0.0.1:3306/test","root","root");}return CONNECTION_INSTANCE;}protected void closeConnection() throws Exception{if(null!=CONNECTION_INSTANCE){if(!CONNECTION_INSTANCE.isClosed()){CONNECTION_INSTANCE.close();}CONNECTION_INSTANCE=null;}}
}

DBUnitUtils.java文件

public class DBUnitUtils {//产生数据集public static void generateDatasetDtd(String[] tables) throws Exception{QueryDataSet dataSet=new QueryDataSet(DButils.getDataBaseConnection());for(String _table:tables){dataSet.addTable(_table);}FlatDtdDataSet.write(dataSet, new FileOutputStream(new File("resource/tmp.dtd")));//FlatXmlDataSet.write(dataSet,new FileOutputStream(new File("resource/dbunit1.xml")));
    }//备份表数据public static void backupDatabase(String[] tables,File backupFile) throws Exception{QueryDataSet dataSet=new QueryDataSet(DButils.getDataBaseConnection());for(String _table:tables){dataSet.addTable(_table);}FlatXmlDataSet.write(dataSet, new FileOutputStream(backupFile));}//清空表数据,并导入测试数据public static void importTables(File dataFile) throws Exception{IDataSet dataSet=new FlatXmlDataSetBuilder().build(dataFile);DatabaseOperation.CLEAN_INSERT.execute(DButils.getDataBaseConnection(), dataSet);}//清空表数据,恢复备份数据public static void resumeDatabase(File backupFile) throws Exception{IDataSet dataSet=new FlatXmlDataSetBuilder().build(backupFile);DatabaseOperation.CLEAN_INSERT.execute(DButils.getDataBaseConnection(), dataSet);        }
}

DButils.java文件

public class DButils {private static IDatabaseConnection conn;//通过dbUnit创建数据库连接public static IDatabaseConnection getDataBaseConnection() throws ClassNotFoundException, SQLException, DatabaseUnitException{if(conn==null){Class.forName("com.mysql.jdbc.Driver");Connection dbConn=DriverManager.getConnection("jdbc:mysql://localhost:3306/test", "root", "root");conn =new MySqlConnection(dbConn,"test");return conn;}return conn;}//关闭数据库连接public static void closeConnection(){if(conn!=null){try {conn.close();} catch (SQLException e) {e.printStackTrace();}}conn=null;}}

User.java文件

public class User {    private long id;private String username;private String password;private String name;public long getId() {return id;}public void setId(long id) {this.id = id;}public String getUsername() {return username;}public void setUsername(String username) {this.username = username;}public String getPassword() {return password;}public void setPassword(String password) {this.password = password;}public String getName() {return name;}public void setName(String name) {this.name = name;}
}

UserService.java文件

public interface UserService {public long create(User user) throws Exception;public void update(User user) throws Exception;public User get(Long id) throws Exception;public void delete(Long id) throws Exception;public List<User> list() throws Exception;
}

UserServiceImpl.java文件

public class UserServiceImpl extends BaseDao implements UserService{//新增用户返回主键public long create(User user) throws Exception {Long id=null;PreparedStatement stm=getConnection().prepareStatement("insert into user(username, password, name) values(?, ?, ?)", Statement.RETURN_GENERATED_KEYS);stm.setString(1, user.getName());stm.setString(2, user.getPassword());stm.setString(3, user.getName());stm.executeUpdate();ResultSet rs=stm.getGeneratedKeys();if(rs.first()){id=rs.getLong(1);}return id;}//更新用户public void update(User user) throws Exception {PreparedStatement stm = getConnection().prepareStatement("update user set password = ?, name = ? where username = ?", Statement.RETURN_GENERATED_KEYS);stm.setString(1, user.getPassword());stm.setString(2, user.getName());stm.setString(3, user.getUsername());stm.executeUpdate();closeConnection();}//获取用户public User get(Long id) throws Exception {User user = null;PreparedStatement stm = getConnection().prepareStatement("select username, password, name from user where id=?");stm.setLong(1, id);ResultSet rs = stm.executeQuery();if (rs.first()) {user = new User();user.setUsername(rs.getString(1));user.setPassword(rs.getString(2));user.setName(rs.getString(3));user.setId(id);}closeConnection();return user;}//删除用户public void delete(Long id) throws Exception {PreparedStatement stm = getConnection().prepareStatement("delete from user where id=?");stm.setLong(1, id);stm.executeUpdate();closeConnection();        }//返回所有用户列表public List<User> list() throws Exception {List<User> users = new ArrayList<User>();PreparedStatement stm = getConnection().prepareStatement("select username, password, name, id from user");ResultSet rs = stm.executeQuery();while (rs.next()) {User user = new User();user.setUsername(rs.getString(1));user.setPassword(rs.getString(2));user.setName(rs.getString(3));user.setId(rs.getLong(4));users.add(user);}closeConnection();return users;}}

UserServiceDBTestCase.java文件

public class UserServiceDBTestCase extends DatabaseTestCase{private UserService userService;@Override//新建一个userDao对象protected void setUp() throws Exception {super.setUp();userService = new UserServiceImpl();}@Override//清空userDao对象protected void tearDown() throws Exception {super.tearDown();userService = null;}@Override//获取数据库连接protected IDatabaseConnection getConnection() throws Exception {Class.forName("com.mysql.jdbc.Driver");Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test", "root", "root");return new MySqlConnection(conn, "test");}//获取输入数据流
    @Overrideprotected IDataSet getDataSet() throws Exception {InputStream is = getClass().getResourceAsStream("/test-data.xml");return new FlatXmlDataSetBuilder().build(is);}//测试数据的创建public void testCreate() throws Exception{User user = new User();user.setName("testdata");user.setPassword("testPassword");user.setUsername("testAdmin");Long id = userService.create(user);assertNotNull(id);User dbUser =userService.get(id);assertNotNull(dbUser);assertEquals("testAdmin", dbUser.getUsername());assertEquals("testPassword", dbUser.getPassword());assertEquals("testAdministrator", dbUser.getName());}public void testGet() throws Exception{User user = userService.get(1L);assertNotNull(user);assertEquals("admin1", user.getUsername());assertEquals("123", user.getPassword());assertEquals("Administrator1", user.getName());User nullUser = userService.get(0L);assertNull(nullUser);}public void testDelete() throws Exception{User user = userService.get(1L);assertNotNull(user);userService.delete(1L);User nullUser = userService.get(1L);assertNull(nullUser);}public void testList() throws Exception{List<User> users = userService.list();assertNotNull(users);assertEquals(4, users.size());}}

UserServiceTestCase.java文件

public class UserServiceTestCase extends TestCase{private UserService userService;    private File backupFile=new File("resource/backup-data.xml");@Overrideprotected void setUp() throws Exception {userService=new UserServiceImpl();}@Beforepublic void setUpData() throws Exception {//备份原来的数据DBUnitUtils.backupDatabase(new String []{"user"}, backupFile);//导入测试数据DBUnitUtils.importTables(new File("resource/test-data.xml"));}@Afterpublic void tearDownData() throws Exception{//恢复备份数据
        DBUnitUtils.resumeDatabase(backupFile);//关闭数据库连接
        DButils.closeConnection();}public void testCreate() throws Exception{DBUnitUtils.importTables(new File("resource/test-data.xml"));User user = new User();DBUnitUtils.resumeDatabase(backupFile);DButils.closeConnection();user.setName("testdata");user.setPassword("testPassword");user.setUsername("testAdmin");Long id = userService.create(user);assertNotNull(id);User dbUser =userService.get(id);assertNotNull(dbUser);assertEquals("testAdmin", dbUser.getUsername());assertEquals("testPassword", dbUser.getPassword());assertEquals("testAdministrator", dbUser.getName());}public void testGet() throws Exception{User user = userService.get(1L);assertNotNull(user);assertEquals("admin1", user.getUsername());assertEquals("123", user.getPassword());assertEquals("Administrator1", user.getName());User nullUser = userService.get(0L);assertNull(nullUser);}public void testDelete() throws Exception{User user = userService.get(1L);assertNotNull(user);userService.delete(1L);User nullUser = userService.get(1L);assertNull(nullUser);}public void testList() throws Exception{        List<User> users = userService.list();assertNotNull(users);assertEquals(4, users.size());}
}

test-data.xml文件

<?xml version='1.0' encoding='UTF-8'?>
<dataset><user id="1" username="test-user1" password="test-user1" name="test-user1"/><user id="2" username="test-user2" password="test-user2" name="test-user2"/>
</dataset>

接下来测试结果: 

demo代码下载: dbunitDemo

总结:
使用了DBUnit后可以实现了对数据库的隔离,成功弥补了JUnit单元测试不清理数据现场的缺憾,实际上DBUnit只是简单的在单元测试前把数据库的数据进行了备份然后插入xml中配置好的数据,在测试结束后再用备份好的原数据库数据填充回数据库.但当面对复杂的表关系和大数据量的时候,每次进行测试都进行数据的备份,也是一个很大的负担,而且把全部的测试数据都编写在xml当中也是很大的工作量

转载于:https://www.cnblogs.com/shawWey/p/10653984.html

DBUnit使用介绍相关推荐

  1. 单元测试之DBUnit的使用以及原理剖析

    前面介绍了不少写单元测试的内容,比方说Mockito和PowerMockito, JUnit 5,经常写单元测试的想必对这些框架都比较熟悉. 这篇博客主要介绍下数据库驱动测试框架–DbUnit(htt ...

  2. 开源单元测试工具汇总

    开源单元测试工具/框架汇总 作为开源软件的爱好者,了解了单元测试这概念后,自然忍不住去找找有什么开源单元测试工具. 下面列出了30款1,不分先后顺序. 1. NoSQL的单元测试工具 NoSQLUni ...

  3. 日志管理(一):slf4j原理简单介绍

    转载自:http://blog.sina.com.cn/s/blog_6f67b91d0100tpqh.html 全称:Simple Logging Facade for Java  简单日志门面(F ...

  4. Android系统源码目录及功能介绍

    Android的移植按如下流程:     1.android linux 内核的普通驱动移植,让内核可以在目标平台上运行起来.     2.正确挂载文件系统,确保内核启动参数和 android 源代码 ...

  5. 简单介绍互联网领域选择与营销方法

    在我看来,互联网领域的选择是"安家",而营销方法的不同则表现了"定家"的方式多种多样,只有选对了,"家"才得以"安定". ...

  6. 常用开源协议介绍以及开源软件规范列表

    1. 开源协议介绍 GPL: General Public License,开源项目最常用的许可证,衍生代码的分发需开源并且也要遵守此协议.该协议也有很多变种,不同变种要求会略微不同. MPL: MP ...

  7. python:Json模块dumps、loads、dump、load介绍

    20210831 https://www.cnblogs.com/bigtreei/p/10466518.html json dump dumps 区别 python:Json模块dumps.load ...

  8. pytorch学习笔记(九):PyTorch结构介绍

    PyTorch结构介绍 对PyTorch架构的粗浅理解,不能保证完全正确,但是希望可以从更高层次上对PyTorch上有个整体把握.水平有限,如有错误,欢迎指错,谢谢! 几个重要的类型 和数值相关的 T ...

  9. Python字节码介绍

    了解 Python 字节码是什么,Python 如何使用它来执行你的代码,以及知道它是如何帮到你的. 如果你曾经编写过 Python,或者只是使用过 Python,你或许经常会看到 Python 源代 ...

最新文章

  1. nginx 之负载均衡 :PHP session 跨多台服务器配置
  2. 泛函分析3——线性空间和赋范线性空间总结
  3. 设计模式(六)J2EE 模式
  4. Tensorflow--图
  5. springBoot微信支付(native)基本使用
  6. 清华EMBA课程系列思考之二 -- 清华探究及信息战略与管理
  7. JSP介绍及视频教程
  8. Win10连接上了wifi但是打开浏览器显示网络异常,诊断网络发现错误“远程计算机或者设备将不接受连接
  9. 服务器带宽超出限制如何解决?
  10. 浅谈Android支付宝快捷支付
  11. 新加坡读计算机专业,【去新加坡读计算机专业】 - 环外新加坡留学网
  12. vue处理PDF文档流数据并实现PDF的预览以及打印功能以及处理PDF打印乱码问题
  13. 常见的五种神经网络(4)-深度信念网络(下)篇之深度信念网络的原理解读、参数学习
  14. 最新版去水印小程序搭建教程
  15. appium安装及使用
  16. 在工业污水处理中实现施耐德PLC的远程监控和上下载
  17. 微软认证考试,还有各种unix认证考试
  18. Framework基础之入门
  19. 51单片机按键控制数码管0~9_(51单片机)课设项目1-按键控制步进电机转向、转速、启停。...
  20. 记自己发现的—SM2国密算法应用的高危漏洞—CVE-2021-3711

热门文章

  1. 计算机事业单位专技岗考什么区别,事业单位管理和专技岗位有什么区别?哪个有前途?...
  2. django重置密码发送html邮件,Django实现发送邮件找回密码功能
  3. 皖西学院微型计算机原理期末试卷,皖西学院 电气微机原理试卷A
  4. java线程 yield_Java线程中yield与join方法的区别
  5. linux中yum怎么安装服务器,yum安装(linux如何安装yum)
  6. 留学申请计算机硕士个人陈述,申请美国计算机专业英文个人陈述
  7. mysql配置参数调优(8GB内存和64GB内存)
  8. 堆栈 cookie 检测代码检测到基于堆栈的缓冲区溢出_WhatsApp缓冲区漏洞曝光 攻击者可通过MP4文件执行远程代码...
  9. python局域网传输文件_Python+pyftpdlib实现局域网文件互传
  10. oracle10grac创建单实例,Oracle10gLinux单实例迁移到rac