Spring+SpringMVC+MyBatis+easyUI整合优化篇(四)单元测试实例
前言
前一篇文章《Spring+SpringMVC+MyBatis+easyUI整合优化篇(三)代码测试》讲了不为和不能两个状态,针对不为,只能自己调整心态了,而对于不能,本文会结合一些实例进行讲解,应该可以使得你掌握单元测试的方法。篇幅所限,所以先写三个类型的测试实例,首先是自己平时写着玩儿的测试类,然后分别是针对数据层和业务层的测试,代码都已经上传到github上了。
我的github地址
简单的测试
我们可能常常会碰到这种事情,需要实现一个功能的时候,忽然想不起来该用什么方法了,或者忽然忘记一个方法该怎么用了,这个时候我可能会查一下API然后写一个简单的测试方法,并没有明确的要去测试什么功能,只是简单的验证一个函数的用法,或者自己实在不确定一个方法该怎么用了,用这种方式加深一下印象。
// 得到MD5加密的内容
@Test
public void md5Test() {
System.out.println(MD5Util.MD5Encode("ssm-maven-secret", "UTF-8"));
//83d8d99f45f62461cc7b7ee76b448cb0
}
// 通过substring()获取文件名
@Test
public void subStringTest() {
//通过substring()获取文件名
String url = "https://s.doubanio.com/f/shire/5522dd1f5b742d1e1394a17f44d590646b63871d/pics/book-default-medium.gif";
url = url.substring(url.lastIndexOf("/") + 1);
System.out.println(url);
//book-default-medium.gif
}
这个只是自己的个人习惯,记忆力有时候真的差。
数据层单元测试
针对书籍模块的测试类,讲解在代码中:
@RunWith(SpringJUnit4ClassRunner.class) //指定测试用例的运行器 这里是指定了Junit4
@ContextConfiguration("classpath:applicationContext.xml")//装配Spring
public class BookDaoTest {
//自动注入,需要将BookDao纳入到Spring容器的管理下,不然会报错
@Autowired
private BookDao bookDao;
@Test
public void getBookByIdTest() {
Book book1 = bookDao.getBookById("1");
Assert.assertEquals(book1, null);//判断两个参数是否相同,返回true的话则测试通过,不然控制台会亮红灯。
Book book2 = bookDao.getBookById("1002");
Assert.assertEquals(book2.getTitle(), "材料成型概论");
// Assert.assertEquals(book2.getTitle(), "我随便写一个");
//写了三个断言,可以分别进行测试,也可以根据上面三个断言自己编写测试。
}
}
针对书籍模块的测试类,讲解也都在代码中:
@RunWith(SpringJUnit4ClassRunner.class) //指定测试用例的运行器 这里是指定了Junit4
@ContextConfiguration("classpath:applicationContext.xml")
@TransactionConfiguration(transactionManager = "transactionManager", defaultRollback = true)
//默认回滚,即此类中的方法即使执行成功,数据也并不会真正的修改,方法执行后会回滚。
//因为对数据库的增删改都会回滚,因此便于测试用例的循环利用
//前面书籍模块的测试由于只有查询方法所以没有加这个注解。
//如果想看到数据库中的数据随着测试而发生变化可以去掉这个注解。
public class UserDaoTest {
@Autowired
//自动注入,需要将BookDao纳入到Spring容器的管理下
private UserDao userDao;
@Test
public void loginTest() {
User user = new User();
user.setUserName("admin");
user.setPassword("123456");
//断言此姓名和密码的用户为空
//密码并没有加密,所以登录失败,返回的用户对象为空。
Assert.assertEquals(userDao.login(user), null);
User user2 = new User();
user2.setUserName("admin");
user2.setPassword(MD5Util.MD5Encode("123456", "UTF-8"));
//断言此姓名和密码的用户可以登录成功,且用户id为2
Assert.assertTrue(userDao.login(user2).getId() == 2);
//执行下面这个断言则会报错。
//Assert.assertTrue(userDao.login(user2).getId() == 3);
}
@Test
public void findUsersTest() {
//断言此时返回的用户列表数大于0
Assert.assertTrue(userDao.findUsers(null).size() > 0);
//断言此时返回的用户列表数等于3,数字你可以随便写,用户数可能是错的,如果报错你会看到控制台一片红色
Assert.assertTrue(userDao.findUsers(null).size() == 3);
}
@Test
public void getTotalUserTest() {
Assert.assertTrue(userDao.getTotalUser(null) > 0);
Assert.assertTrue(userDao.getTotalUser(null) == 3);
}
//使用update、insert、delete方法时,会得到一个返回值,这个返回值说明了执行一条sql语句后,表中有多少条记录被影响了。
//比如用update修改一条记录,如果修改成功了,返回值为1,返回0则是修改失败。
@Test
public void updateUserTest() {
User user = new User();
user.setId(51);
user.setPassword("1221");
//大于0的意思是成功修改了一条记录,即修改成功,如果updateUser()方法返回值等于0,即修改失败
Assert.assertTrue(userDao.updateUser(user) > 0);
User user2 = new User();
user2.setId(1000);
user2.setPassword("234y9823y89hhao");
Assert.assertTrue(userDao.updateUser(user2) > 0);
}
@Test
public void addUserTest() {
User user = new User();
user.setUserName("测试用户");
user.setPassword(MD5Util.MD5Encode("testuser", "UTF-8"));
//大于0的意思是影响了数据库中的一条记录,即添加成功
Assert.assertTrue(userDao.addUser(user) > 0);
}
@Test
public void deleteUserTest() {
Assert.assertTrue(userDao.deleteUser(51) > 0);
}
}
大家可以将代码pull到本地进行测试,并根据上面几个方法写一下自己的测试用例。
service的单元测试
其实service层的单元测试和dao层并没有太多区别,唯一的不同可能就是service层方法中可以执行多条sql语句,而dao层的方法只能执行一条sql语句,因此这个例子更侧重于讲一下事务问题。
配置文件:
<!-- 配置事务通知属性 -->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<!-- 定义事务传播属性 -->
<tx:attributes>
<!--事务切面中insert开头的方法会被纳入事务管理中-->
<tx:method name="insert*" propagation="REQUIRED"/>
<tx:method name="update*" propagation="REQUIRED"/>
<tx:method name="upd*" propagation="REQUIRED"/>
</tx:attributes>
......
</tx:advice>
<!-- 配置事务切面 -->
<aop:config>
<aop:pointcut id="serviceOperation"
expression="(execution(* com.ssm.maven.core.service.*.*(..)))"/>
<aop:advisor advice-ref="txAdvice" pointcut-ref="serviceOperation"/>
</aop:config>
业务方法:
//通过Spring的配置文件,此方法已经纳入到其事务管理下
//发生异常时会触发事务回滚,数据不会被更改
public int insertStore(Store store) {
int level = Integer.valueOf(store.getLevel());
for (int i = 1; i < level; i++) {
store.setLevel(i + "");
storeDao.insertStore(store);
}
store.setLevel(level + "");
int result = storeDao.insertStore(store);
int i = 10 / 0;
//发生异常,操作回滚.
//可以试着将上面一条语句注释掉再运行测试用例,看看有什么区别。
return result;
}
测试用例:
@RunWith(SpringJUnit4ClassRunner.class) //指定测试用例的运行器 这里是指定了Junit4
@ContextConfiguration("classpath:applicationContext.xml")
//@TransactionConfiguration(transactionManager = "transactionManager", defaultRollback = true)
//不添加此设置,测试service层的事务管理
//service层与dao层的测试时相同的,不同之处,在于service层多数都会在配置文件中配置spring的事务管理
public class StoreServiceTest {
@Autowired
private StoreService storeService;
@Test
public void insertStoreTest() {
Store store = new Store();
store.setLevel("5");
store.setNumber("TEST");
storeService.insertStore(store);
int i = 10 / 0;
//这里发生异常是不会回滚的,因为此方法并没有被纳入事务管理中
}
}
这个例子可能有些简单,一般业务层方法都会较为复杂,如下:
function(){
-A-
SQL1
-B-
SQL2
-C-
SQL3
-D-
}
那么执行此方法时,无论在A处、B处、C处、D处发生异常,都会触发事务回滚,关于Spring的事务管理及事务传播属性,想了解的可以自己去查询一下。
总结
本篇主要讲了数据层和业务层的测试,下一篇主要会讲一下controller控制层的测试,睡觉啦...
Spring+SpringMVC+MyBatis+easyUI整合优化篇(四)单元测试实例的更多相关文章
- Spring+SpringMVC+MyBatis+easyUI整合优化篇(二)Log4j讲解与整合
日常啰嗦 上一篇文章主要讲述了一下syso和Log间的一些区别与比较,重点是在项目的日志功能上,因此,承接前文<Spring+SpringMVC+MyBatis+easyUI整合优化篇(一)Sy ...
- Spring+SpringMVC+MyBatis+easyUI整合优化篇(五)结合MockMvc进行服务端的单元测试
日常啰嗦 承接前一篇文章<Spring+SpringMVC+MyBatis+easyUI整合优化篇(四)单元测试实例>,已经讲解了dao层和service层的单元测试,还有控制器这层也不能 ...
- Spring+SpringMVC+MyBatis+easyUI整合优化篇(十三)数据层优化-表规范、索引优化
本文提要 最近写的几篇文章都是关于数据层优化方面的,这几天也在想还有哪些地方可以优化改进,结合日志和项目代码发现,关于数据层的优化,还是有几个方面可以继续修改的,代码方面,整合了druid数据源也开启 ...
- Spring+SpringMVC+MyBatis+easyUI整合优化篇
优化篇 Spring+SpringMVC+MyBatis+easyUI整合优化篇(一)System.out.print与Log Spring+SpringMVC+MyBatis+easyUI整合优化篇 ...
- Spring+SpringMVC+MyBatis+easyUI整合优化篇(七)图片上传功能
日常啰嗦 前一篇文章<Spring+SpringMVC+MyBatis+easyUI整合优化篇(六)easyUI与富文本编辑器UEditor整合>讲了富文本编辑器UEditor的整合与使用 ...
- Spring+SpringMVC+MyBatis+easyUI整合优化篇(十四)谈谈写博客的原因和项目优化
阶段总结 又到了优化篇的收尾阶段了,这其实是一篇阶段总结性的文章,今天是4月29号,距离第一次发布博客已经两个月零5天,这两个多月的时间,完成了第一个项目ssm-demo的更新,过程中也写了33篇博客 ...
- Spring+SpringMVC+MyBatis+easyUI整合优化篇(六)easyUI与富文本编辑器UEditor整合
日常啰嗦 本来这一篇和接下来的几篇是打算讲一下JDBC和数据库优化的,但是最近很多朋友加我好友也讨论了一些问题,我发现大家似乎都是拿这个项目作为练手项目,作为脚手架来用的,因此呢,改变了一下思路,JD ...
- Spring+SpringMVC+MyBatis+easyUI整合优化篇(一)System.out.print与Log
日常啰嗦 距离上一次更新博客有一段时间了,主要是因为最近有开发任务,另外,这段时间也在学习docker的相关知识,所以博客就没有继续写了,推荐一本书<Docker技术入门与实战>(第二版) ...
- Spring+SpringMVC+MyBatis+easyUI整合优化篇(一)Java语言中System.out.print与Log的比较
作者:13 GitHub:https://github.com/ZHENFENG13 版权声明:本文为原创文章,未经允许不得转载. 前言 距离上一次更新博客有一段时间了,主要是因为最近有开发任务,另外 ...
随机推荐
- 关于nodejs express4.X框架不支持layout模板的问题解决
网上有有种方法是安装express-partials模块,然后在 app.set(‘view engine’, ‘ejs’); 这句后面加上app.use(partials());但是,经过我的反复尝 ...
- bootstrap-导航总结
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- MySQL日志系统
body { font-family: Helvetica, arial, sans-serif; font-size: 14px; line-height: 1.6; padding-top: 10 ...
- JavaScript实现按键精灵
最近有个需求,需要在页面上面自动点击.输入.提交. 用以模拟真实用户的操作行为,可以通过直接执行某个元素绑定的事件,来执行操作. 也可以创建事件,再派发事件,执行操作.关于事件的更多细节,可以参考&l ...
- 2月22日 《从Paxos到Zookeeper 分布式一致性原理与实践》读后感
zk的特点: 分布式一致性的解决方案,包括:顺序一致性,原子性,单一视图,可靠性,实时性 zk的基本概念: 集群角色:not Master/Slave,is Leader/Follower/Obser ...
- 【转载】简析TCP的三次握手与四次分手
最近在补习HTTP协议相关知识点,看到这篇讲得不错,所以转载收藏一下,同时也分享给大家.原文地址:http://www.jellythink.com/archives/705,版权归原作者所有. TC ...
- centos快速安装redis
mkdir redis cd redis wget http://labfile.oss.aliyuncs.com/files0422/redis-2.8.9.tar.gz 解压 tar -xvfz ...
- .NET平台和开发.
- iOS集成ApplePay
Apple Pay正式在国内上线的那天,一起工作的小伙伴就走进了Starbucks,7-11等带有银联闪付的店进行了尝鲜.不管是否要再次输入一次密码,但是它的出现确实给我们带来了极大的便捷.下面就尝试 ...
- ArrayList源码剖析
ArrayList简介 ArrayList是基于数组实现的,是一个动态数组,其容量能自动增长,类似于C语言中的动态申请内存,动态增长内存. ArrayList不是线程安全的,只能用在单线程环境下,多线 ...