JMockit学习笔记
一、在eclipse下建立JMockit工程
1、下载最新版JMockit(jmockit-1.4.zip);
2、解压后的文件夹包含有:library jars, source files, samples, API javadocs, and additional documentation;
3、将jmockit.jar添加到项目classpath中;
特别的:
1)确保classpath中Jar包的顺序:jmockit的jar包必须在junit之前(通过Order and Export" 标签上下移动);
2)eclipse项目所用JRE来自于JDK,而不是“简洁”版的JRE,因为后者缺少本地类库“attach”。

二、行为和状态的测试
基于行为(Behavior-based)的mock是站在目标测试代码外面的角度的,通常主要模拟行为,而基于状态
(State-based)的是站在目标测试代码内部的。我们可以对传入的参数进行检查、匹配,才返回某些结果。
Mockup用于State-based测试。
二、声明和使用mock类型
1、字段,期望块的字段与期望块内的局部属性字段使用@Mocked来声明Mock类型。
2、参数,方法的参数声明来引入一个Mock类型。
第一种情况,属性字段是属于测试类或者一个mockit.Expectations子类(一个expectation期望块的内部的局部
属性字段)。
第二种情况,参数必须是属于某个测试方法(@Test标签下的方法)。
在所有的情况,一个mock属性字段或者参数声明,都可以通过使用@Mocked声明。对于方法mock的参数或者
在expectations期望块中定义的mock属性字段来说,该注解是可选的,而对于定义在测试类(XXXTest类)中的
属性字段,@Mocked标签是必须,这是为了防止和该测试类的其它不需要mock的字段属性产生冲突。
package main;
import static org.junit.Assert.*;
import java.io.Serializable;
import org.junit.Test;
import mockit.Expectations;
import mockit.Mocked;
import mockit.NonStrictExpectations;
import mockit.Verifications;
/*
* 一个用户接口(外部依赖)
*/
interface Dependency {
String doSomething(boolean b);
}
// 声明变量类型MultiMock(它实现两个接口,作用域是整个测试类)
public class MultiMocksTest<MultiMock extends Dependency & Runnable> {
@Mocked
MultiMock multiMock;
@Test
public void mockFieldWithTwoInterfaces() {
new NonStrictExpectations() {
{
multiMock.doSomething(false);
result = "test";
}
};
multiMock.run();
assertEquals("test", multiMock.doSomething(false));
// 验证run()方法执行一次
new Verifications() {
{
multiMock.run();
}
};
}
@Test
// 声明变量类型M,它实现两个接口,作用域为该测试方法
// final M mock 前的@Mocked注解是可选的
public <M extends Dependency & Serializable> void mockParameterWithTwoInterfaces(
@Mocked final M mock) {
new Expectations() {
{
mock.doSomething(true);
result = "";
}
};
assertEquals("", multiMock.doSomething(true));
}
}
3、对于一个返回值不为void类型的方法,Expectations中如何模拟方法返回值:
1)其返回值可以通过Expectations的result属性域来记录
2)Expectations的returns(Object)方法来记录
例如,方法返回一个Throwable异常类,只需将一个类型实验赋给result(注意,异常类只能通过result方式
赋值)。
package main;
import mockit.Expectations;
import org.junit.Test;
public class UnitUnderTest {
// 1、构造方法
private OutWork work = new OutWork();
public void doSomthing() {
// 2、intReturningMethod()方法
int n = work.intReturningMethod();
for (int i = 0; i < n; i++) {
String s;
try {
// 3、stringReturningMethod()方法
s = work.stringReturningMethod();
} catch (Exception e) {
// 处理异常
e.printStackTrace();
}
}
// 其它逻辑...
}
@Test
public void test() {
new Expectations() {
OutWork work;
{
// 1、构造方法模拟
new OutWork();
// 2、intReturningMethod()方法模拟
work.intReturningMethod();
result = 3;
// 3、stringReturningMethod()方法模拟
work.stringReturningMethod();
// 方法分别返回三个值,两个字符串,一个异常
returns("str1", "str2");
result = new Exception("testException");
}
};
new UnitUnderTest().doSomthing();
}
}
class OutWork {
public int intReturningMethod() {
return 0;
}
public String stringReturningMethod() {
return "";
}
}
三、从严格到非严格
1、@Mocked+Expections块:会进行隐式校验(执行顺序和次数)
2、@Mocked+NonStrictExpections块:NonStrictExpections块中的Incovation可以非严格执行(不执行或者
执行N次,除非显示地指定执行次数)。
package main;
/*
* 用于Mock的接口
*/
public interface WinportUrlService {
public boolean hasWinport(String id);
public String getMsg();
public Throwable getWinportUrlThrowException(String id);
}
package main;
import static org.junit.Assert.*;
import mockit.Expectations;
import mockit.Mocked;
import mockit.NonStrictExpectations;
import org.junit.Test;
public class IntroductionTest {
// @Mocked注解+Expectations/NonStrictExpectations块
@Mocked
private WinportUrlService winportUrlService = null;
@Test
public void testNoExceptions() {
final String memberId = "test";
// 未指定期望块,方法返回默认值
assertEquals(false, winportUrlService.hasWinport(memberId));
assertEquals(null, winportUrlService.getMsg());
assertEquals(null, winportUrlService.getWinportUrlThrowException(memberId));
}
@Test
public void testWithExpectations() {
final String memerId = "test";
// 步骤一:record
// 严格期望块
new Expectations() {
{
// 下面的Invocation必须严格执行
winportUrlService.hasWinport(memerId);
result = false;// 也可以是returns(false)
// 未指定执行次数
}
};
// 非严格期望块
new NonStrictExpectations() {
{
// 下面的Invocation非严格执行
winportUrlService.getMsg();
result = "test";// 也可以是returns("test")
// 未指定执行次数
}
};
// 步骤二:replay阶段
// hasWinport必须严格执行一次
assertEquals(false, winportUrlService.hasWinport(memerId));
// getMsg可以不执行或执行多次
assertEquals("test", winportUrlService.getMsg());
assertEquals("test", winportUrlService.getMsg());
// 下面的Invocation失败
// winportUrlService.getWinportUrlThrowException(memerId);
}
}
3、@NonStrict:可以在replay中调用或不调用。@NonStrict可以避免需要记录调用构造函数,或任何不感兴
趣的方法。
注意:@NonStrict它是针对类的属性非严格,类的属性适用于类中的所有测试方法。这个非严格的范围比
NonStrictExpections块的作用范围大很多,一旦使用了@NonStrict,Expections中的Incovation就变成了非严
格的invocation,因此其它测试方法还想在该属性的基础上使用Expections块是不可能的了。如果是这种情况
就需要这样使用(@Mocked+NonStrictExpectations块)。
package main;
import static org.junit.Assert.*;
import mockit.Expectations;
import mockit.NonStrict;
import org.junit.Test;
public class IntroductionTest2 {
// @NonStrict注解
@NonStrict
private WinportUrlService winportUrlService = null;
@Test
public void testWithExpectations() {
final String memberId = "test";
// 步骤一:record
// 用了@NonStrict,Expectation中的invocation就变成了非严格的invocation
new Expectations() {
{
// 下面的Invocation非严格执行
winportUrlService.hasWinport(memberId);
result = false;// 也可以是returns(false)
// 未指定执行次数
}
};
// 步骤二:replay阶段
// hasWinport可以不执行或执行多次
assertEquals(false, winportUrlService.hasWinport(memberId));
assertEquals(false, winportUrlService.hasWinport(memberId));
// 下面的Invocation成功
winportUrlService.getWinportUrlThrowException(memberId);
}
}
4、此外,若不指定执行次数,Expections块默认必须执行一次,NonStrictExpections块中的Invocation可执行
N次或不执行;若显式地指定执行次数(N次),二者的Invocation都必须执行N次
JMockit学习笔记的更多相关文章
- js学习笔记:webpack基础入门(一)
之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...
- PHP-自定义模板-学习笔记
1. 开始 这几天,看了李炎恢老师的<PHP第二季度视频>中的“章节7:创建TPL自定义模板”,做一个学习笔记,通过绘制架构图.UML类图和思维导图,来对加深理解. 2. 整体架构图 ...
- PHP-会员登录与注册例子解析-学习笔记
1.开始 最近开始学习李炎恢老师的<PHP第二季度视频>中的“章节5:使用OOP注册会员”,做一个学习笔记,通过绘制基本页面流程和UML类图,来对加深理解. 2.基本页面流程 3.通过UM ...
- 2014年暑假c#学习笔记目录
2014年暑假c#学习笔记 一.C#编程基础 1. c#编程基础之枚举 2. c#编程基础之函数可变参数 3. c#编程基础之字符串基础 4. c#编程基础之字符串函数 5.c#编程基础之ref.ou ...
- JAVA GUI编程学习笔记目录
2014年暑假JAVA GUI编程学习笔记目录 1.JAVA之GUI编程概述 2.JAVA之GUI编程布局 3.JAVA之GUI编程Frame窗口 4.JAVA之GUI编程事件监听机制 5.JAVA之 ...
- seaJs学习笔记2 – seaJs组建库的使用
原文地址:seaJs学习笔记2 – seaJs组建库的使用 我觉得学习新东西并不是会使用它就够了的,会使用仅仅代表你看懂了,理解了,二不代表你深入了,彻悟了它的精髓. 所以不断的学习将是源源不断. 最 ...
- CSS学习笔记
CSS学习笔记 2016年12月15日整理 CSS基础 Chapter1 在console输入escape("宋体") ENTER 就会出现unicode编码 显示"%u ...
- HTML学习笔记
HTML学习笔记 2016年12月15日整理 Chapter1 URL(scheme://host.domain:port/path/filename) scheme: 定义因特网服务的类型,常见的为 ...
- DirectX Graphics Infrastructure(DXGI):最佳范例 学习笔记
今天要学习的这篇文章写的算是比较早的了,大概在DX11时代就写好了,当时龙书11版看得很潦草,并没有注意这篇文章,现在看12,觉得是跳不过去的一篇文章,地址如下: https://msdn.micro ...
随机推荐
- verilog 常见单元描述
半加器: //行为级建模 module half_adder2(a, b, sum, c_out); input a, b; output sum, c_out; assign {c_out, sum ...
- 《anchor-based v.s. anchor-free》
作者:青青子衿链接:https://www.zhihu.com/question/356551927/answer/926659692来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载 ...
- LG5536 「XR-3」核心城市 树的直径
问题描述 LG5536 题解 两次 \(\mathrm{dfs}\) 求树的直径. 然后找到树的直径的中点. 然后按照 子树中最深的点深度-自己深度 排序,贪心选取前 \(k\) 个. \(\math ...
- 基于C++的STL的vector实现静态链表,要求包含插入,删除,和查找功能
//main.cpp部分 #include"List.cpp" int main() { StaticList<int> SL; SL.Insert(,); SL.In ...
- Linux 操作MySQL常用命令行
1.连接数据库 mysql -uroot -p Enter password: ** Mysql> 出现mysql>说明成功连接到数据 2.显示数据库 mysql> show dat ...
- Python高级应用程序设计任务期末作业
Python高级应用程序设计任务要求 用Python实现一个面向主题的网络爬虫程序,并完成以下内容:(注:每人一题,主题内容自选,所有设计内容与源代码需提交到博客园平台) 一.主题式网络爬虫设计方案( ...
- 优雅的阅读CSDN博客
CSDN现在似乎不强制登录了2333.但是广告多了也是碍眼的不行...将下列css添加到stylus中就行了. 代码转自xzz的博客. 自己修改了一下,屏蔽了登录弹出框. .article_conte ...
- zlib压缩相关
相关原理 deflate(RFC1951):一种压缩算法,使用LZ77和哈弗曼进行编码: zlib(RFC1950):一种格式,是对deflate进行了简单的封装,他也是一个实现库(delphi中有z ...
- Python连载41-yield from详解、委派生成器
一. 1.yield from (1)调用协程为了得到返回值,协程必须正常终止 (2)生成器正常终止会发出StopIteration异常,异常对象的value属性保存返回值. (3)yield fro ...
- ElementUI中如何实现Form表单内的文字居中
<el-table :data='orderList' border stripe :align='center' :cell-style='cellStyle' :header-cell-st ...