Mockito 是当前最流行的 Java 单元测试 Mock 框架,JMockit天然支持静态方法和构造函数的 Mock,到底哪个更好用呢?
Mock 介绍
为什么要使用 mock
当我们写单元测试时,我们往往只想验证我们所写函数的功能,而不是它的依赖项。但是有时候它的依赖项并不可控。
为了把函数的依赖项剥离,我们就需要为此依赖项提供一个替代品。通过这种方式,我们可以强制依赖项返回特定值,抛出异常,或者将比较耗时的方法减少到固定的值。
这种替代品就是 mock,它可以帮我们简化测试编码并减少测试执行时间。
到底要不要 mock
不是所有的东西都要被 mock 的。有时候如果 mock 带来的好处并不明显,我们该考虑的是是不是换成集成测试更合理等,而不是强行 mock。
测试用例
假设我们有这样一个场景:我们对外提供了一个保存 toDo list 的服务,用户每次访问接口的时候传递 toDo 编号及 toDo 详情信息,然后我们调用 ToDoService 来处理逻辑,在实际的处理中我们需要调用 DAO 层来保存数据。
编码实现
首先我们有一个 ToDo 的实体类。
public class ToDoModel {
private Long numbering;
private String toDoDetail;
}
在 ToDoDao 里面,我们有一个 save 方法,但是我们不需要具体实现它,我们后面的例子会对它进行 mock。
public class ToDoDao {
public int save(ToDoModel toDoModel) {
return 0;
}
}
在 ToDoService 里,我们同样实现 save 方法,在 save 方法里会调用 Dao 层的 save 方法。且当我们配置了环境变量STOP_SERVICE为 true 后,我们默认返回保存失败,不进行保存。我们再提供一个返回 void 的 setCurrentNumbering 方法,后面我们测试会用到。
public class ToDoService {
private ToDoDao toDoDao;
private long currentNumbering;
public boolean save(ToDoModel toDoModel) {
assert toDoModel != null;
if (Boolean.parseBoolean(Optional.ofNullable(System.getenv("STOP_SERVICE")).orElse("false"))) {
return false;
}
int result = toDoDao.save(toDoModel);
switch (result) {
case 1:
return true;
default:
return false;
}
}
public void setCurrentNumbering(long numbering) {
if (numbering > 0) {
this.currentNumbering = numbering;
}
}
}
最后,ToDoController 里将调用 ToDoService 方法。
public class ToDoController {
public ToDoService toDoService;
public String add(ToDoModel toDoModel) {
if (toDoModel == null) {
return "ERROR";
} else {
boolean added;
try {
added = toDoService.save(toDoModel);
} catch (Exception e) {
return "ERROR";
}
if (added) {
toDoService.setCurrentNumbering(toDoModel.getNumbering());
return "OK";
} else {
return "NOT OK";
}
}
}
}
当前,我们已经有了一些逻辑代码,下面我们分别使用 Mockito 以及 JMockit 来对他们进行一个 Mock 测试。
测试设置
Mock
相关链接
发表评论