jmockit使用总结-MockUp重点介绍
公司对开发人员的单元测试要求比较高,要求分支覆盖率、行覆盖率等要达到60%以上等等。项目中已经集成了jmockit这个功能强大的mock框架,学会使用这个框架势在必行。从第一次写一点不会,到完全可以应付工作要求,期间踩了好多坑,学到了不少东西。下面简单总结一下jmockit这个框架的使用,重点介绍MockUp的使用,因为项目中都采用此种方式模拟方法。
一、框架集成
添加maven依赖
<dependencies>
<!-- jmockit必须写在junit之前 -->
<dependency>
<groupId>org.jmockit</groupId>
<artifactId>jmockit</artifactId>
<version>1.16</version>
<scope>test</scope>
</dependency> <dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
</dependency> </dependencies>
二、@Mocked模拟方式介绍
@Mocked模拟,由录制、回放、验证三步骤完成,是对某个类的所有实例的所有方法进行完整的模拟方式。
/**
* 被测试类
*/
public class App { public String say() {
return "Hello World";
} public String say2(){
return "Hello World 2";
} public static String staticSay() {
return "Still hello world";
}
}
/**
* 测试类
*/
public class AppTest { /**
* 针对类及所有实例的的整体模拟,未写录制的方法默认返回0,null等
*/
@Mocked
App app; @Test
public void testSay() { //录制,定义被模拟的方法的返回值,可以录制多个行为,写在一个大括号里也可以,多个大括号隔开也可以
new Expectations() {{
app.say();
result = "say";
}}; //回放,调用模拟的方法
System.out.println(app.say()); //say
System.out.println(new App().say()); //say
System.out.println(App.staticSay()); //null //验证
new Verifications() {{
//验证say模拟方法被调用,且调用了2次
app.say();
times = 2; //验证staticSay模拟方法被调用,且调用了1次
App.staticSay();
times = 1;
}}; }
}
三、@Injectable模拟方式介绍
@Injectable和@Mocked的方式很像,区别是@Injectable仅仅对当前实例进行模拟。
/**
* 测试类
*/
public class AppTest001 { /**
* 仅针对当前实例的整体模拟
*/
@Injectable
App app; @Test
public void testSay() { //录制
new Expectations() {{
app.say();
result = "say";
}}; //回放
System.out.println(app.say()); //say,模拟值
System.out.println(app.say2()); //null,模拟默认值 final App appNew = new App();
System.out.println(appNew.say()); //Hello World,未被模拟,方法实际值
System.out.println(App.staticSay()); //Still hello world,未被模拟,方法实际值 //验证
new Verifications() {
{
//验证say模拟方法被调用
app.say();
times = 1;
}
{
appNew.say();
times = 1;
}
{
//验证staticSay模拟方法被调用,且调用了1次
App.staticSay();
times = 1;
}
}; }
}
四、Expectations传参,局部模拟
/**
* 测试类
*/
public class AppTest002 { @Test
public void testSay() { final App app = new App(); //录制,带参数表示局部模拟【针对所有实例的局部模拟,具体到某个方法的模拟,其它方法不模拟】
new Expectations(App.class) {{
app.say();
result = "say";
}}; //回放
System.out.println(app.say()); //say,模拟值
System.out.println(app.say2()); //Hello World 2 ,未被模拟,方法实际值
System.out.println(new App().say()); //say,模拟值
System.out.println(App.staticSay()); //Still hello world,未被模拟,方法实际值 }
}
五、MockUp局部模拟,可重写原有方法的逻辑,比较灵活,推荐使用
/**
* 测试类
*/
public class AppTest003 { @Test
public void testSay() {
//局部模拟【针对所有实例的局部模拟,具体到某个方法的模拟,其它方法不模拟】
new MockUp<App>(App.class){ @Mock
String say(){
return "say";
}
}; //回放
System.out.println(new App().say()); //say,模拟值
System.out.println(new App().say2()); //Hello World 2,未被模拟,方法实际值
System.out.println(App.staticSay()); //Still hello world,未被模拟,方法实际值 }
}
六、MockUp如何模拟私有方法、静态方法、静态块、构造函数等
1.模拟私有属性(实例属性和类属性),MockUp不支持,采用如下方式
//模拟实例的字段
Deencapsulation.setField(Object objectWithField, String fieldName, Object fieldValue) //模拟类的静态字段
Deencapsulation.setField(Class<?> classWithStaticField, String fieldName, Object fieldValue)
2.模拟私有方法,MockUp不支持,采用如下方式
//模拟实例方法,注意参数不能为null,如果要传null请使用带参数类型的另一个重载方法
Deencapsulation.invoke(Object objectWithMethod, String methodName, Object... nonNullArgs) //模拟类方法
Deencapsulation.invoke(Class<?> classWithStaticMethod, String methodName, Object... nonNullArgs)
3.模拟静态方法
和模拟实例方法一样,去掉static即可
4.模拟静态块
//mock静态代码块
@Mock
void $clinit(Invocation invocation){ }
5.模拟实例块和构造函数
//mock代码块和构造函数
@Mock
void $init(Invocation invocation) { }
6.模拟接口,MockUp不支持,采用@Capturing
/**
* 被模拟的接口
*/
public interface IUserService { String getUserName( );
}
public class UserServiceImpl implements IUserService {
@Override
public String getUserName() {
return "Bob";
}
}
/**
* 接口模拟测试
*/
public class IUserServiceTest { /**
* MockUp不能mock接口方法,可以用来生成接口实例
*/
@Test
public void getUserNameTest001(){
MockUp<IUserService> mockUp = new MockUp<IUserService>(){ @Mock
String getUserName( ){
return "Jack";
}
}; IUserService obj = new UserServiceImpl();
System.out.println(obj.getUserName()); //Bob,mock失败 obj = mockUp.getMockInstance();
System.out.println(obj.getUserName()); //Jack,mockUp生成的实例,和自己写一个接口实现一样 obj = new UserServiceImpl();
System.out.println(obj.getUserName()); //Bob,mock失败
} /**
* @Capturing 注解可以实现mock接口,所有实现类的实例均被mock
* @param base
*/
@Test
public void getUserNameTest002(@Capturing final IUserService base){
IUserService obj = new UserServiceImpl();
System.out.println(obj.getUserName()); //mock成功,返回模拟默认值null //录制
new Expectations(){
{
base.getUserName();
result = "Jack";
}
};
System.out.println(obj.getUserName()); //Jack obj = new IUserService() {
@Override
public String getUserName() {
return "Alice";
}
};
System.out.println(obj.getUserName()); //Jack
} }
七、MockUp模拟方法中调用原方法
/**
* 模拟方法调用原方法逻辑测试
*/
public class JSONObjectTest { @Test
public void getTest(){
JSONObject jsonObject = new JSONObject();
jsonObject.put("a","A");
jsonObject.put("b","B");
jsonObject.put("c","C");
System.out.println(jsonObject.get("a")); //A
System.out.println(jsonObject.get("b")); //B
System.out.println(jsonObject.get("c")); //C
new MockUp<JSONObject>(){
@Mock
Object get(Invocation invocation,Object key){
if("a".equals(key)){
return "aa";
}else{
//调用原逻辑
return invocation.proceed(key);
}
}
};
System.out.println(jsonObject.get("a")); //aa
System.out.println(jsonObject.get("b")); //B
System.out.println(jsonObject.get("c")); //C
}
}
八、MockUp单元测试用例单个跑正常,批量跑失败可能的原因
1.测试方法使用了共享变量,相互影响。
2.在一个测试方法里多次MockUp同一个类,将某个类需要mock的方法均写在一个new MockUp里即可解决,原因未知。
3.在测试方法里的MockUp,在方法结束前,调用一下mockUp.tearDown。
jmockit使用总结-MockUp重点介绍的更多相关文章
- 消息队列介绍、RabbitMQ&Redis的重点介绍与简单应用
消息队列介绍.RabbitMQ&Redis的重点介绍与简单应用 消息队列介绍.RabbitMQ.Redis 一.什么是消息队列 这个概念我们百度Google能查到一大堆文章,所以我就通俗的讲下 ...
- .NET:“事务、并发、并发问题、事务隔离级别、锁”小议,重点介绍:“事务隔离级别"如何影响 “锁”?
备注 我们知道事务的重要性,我们同样知道系统会出现并发,而且,一直在准求高并发,但是多数新手(包括我自己)经常忽略并发问题(更新丢失.脏读.不可重复读.幻读),如何应对并发问题呢?和线程并发控制一样, ...
- 各种vpn协议介绍(重点介绍sslvpn的实现方式openvpn)
vpn介绍: VIrtual Private Network 虚拟专用网络哪些用户会用vpn? 公司的远程用户(出差.家里),公司的分支机构.idc机房.企业间.FQ常见vpn协议有哪些? ...
- spring框架总结(03)重点介绍(Spring框架的第二种核心掌握)
1.Spring的AOP编程 什么是AOP? ----- 在软件行业AOP为Aspect Oriented Programming 也就是面向切面编程,使用AOP编程的好处就是:在不修改源代码的情 ...
- 进击的Python【第十一章】:消息队列介绍、RabbitMQ&Redis的重点介绍与简单应用
消息队列介绍.RabbitMQ.Redis 一.什么是消息队列 这个概念我们百度Google能查到一大堆文章,所以我就通俗的讲下消息队列的基本思路. 还记得原来写过Queue的文章,不管是线程queu ...
- java中关于IO流的知识总结(重点介绍文件流的使用)
今天做软件构造实验一的时候,用到了java文件流的使用,因为之前学的不是很踏实,于是决定今天好好总结一下, 也方便以后的回顾. 首先,要分清IO流中的一些基础划分: 按照数据流的方向不同可以分为:输入 ...
- html的特质语义:微格式及其他(重点介绍其中两种)
今天再次翻开html的书本, 感觉过了个周末似乎生疏了许多, 虽然我是刚接触html的, 但是对于他还是抱有极其大的兴趣的, 所以不爱看书的我, 也开始一遍遍的翻阅着书本, 寻找解决问题的方法, 下面 ...
- ISO c++ 14 重点介绍[译]
原文链接 http://marknelson.us/2014/09/11/highlights-of-iso-c14/ 下面是对你的日常开发有重大影响的C++14新变动,列出了一些示例代码,并讨论何时 ...
- 谈谈javascript数组排序方法sort()的使用,重点介绍参数使用及内部机制?
语法:arrayObject.sort(sortby) 参数sortby可选,规定排序顺序,必须是函数: 注:如果调用该方法时没有使用参数,将按字符编码的顺序进行排序,要实现这一点,首先应把数组的元素 ...
随机推荐
- 第五篇 -- Xml序列化
XML序列化是将对象的公共属性和字段转换为XML格式,以便存储或传输的过程.反序列化则是从XML输出中重新创建原始状态的对象.XML序列化中最主要的类是XmlSerializer类.它的最重要的方法是 ...
- 20180530模拟赛T2——绀碧之棺
题目背景 qiancl 得到了一张藏宝图,上面写了一道谜题. 题目描述 定义\(F(n)\)为 n 在十进制下各个数位的平方和,求区间\([a,b]\)中有多少\(n\)满足\(k\times F(n ...
- reduce要素与适用总结
要素: 1.高阶函数:reduce: 2.处理函数:reducer: 3.数据:可以是具体数据.签名相同的普通函数.签名相同的高阶函数: reduce(reducer, datas(data or f ...
- RxSwift 在本质上简化了开发异步程序
RxSwift 是一个组合异步和事件驱动编程的库,通过使用可观察序列和功能样式运算符来,从而允许通过调度程序进行参数化执行. RxSwift 在本质上简化了开发异步程序,允许代码对新数据作出反应,并以 ...
- linux MySQL5.7 rpm安装(转)
删除旧包: # rpm -qa | grep -i mysql # rpm -ev mysql-libs-* --nodeps 安装rpm包: # rpm -ivh mysql-community-c ...
- 使用Google学术简单方法汇总
1 Google学术打不开,简单方法汇总. 2 谷歌学术镜像 http://dir.scmor.com/google/ 3,https://xs.glgoo.net/ 4, https://sch ...
- space-cloud 学习一 基本试用
space-cloud 是一个支持多数据库,以下是一个简单的基于官方文档的试用 使用docker-compose 运行 环境准备 下载docker-compose文件 wget https://raw ...
- s3-sftp-proxy goreleaser rpm &&deb 包制作
上次写过简单的s3-sftp-proxy基于容器构建以及使用goreleaser构建跨平台二进制文件的,下边演示下关于 rpm&&deb 包的制作,我们只需要简单的配置就可以生成方便安 ...
- A1102 | 反转二叉树
#include <stdio.h> #include <memory.h> #include <math.h> #include <string> # ...
- 虔诚的墓主人(BZOJ1227)(洛谷P2154)解题报告
题目描述 小W是一片新造公墓的管理人.公墓可以看成一块N×M的矩形,矩形的每个格点,要么种着一棵常青树,要么是一块还没有归属的墓地. 当地的居民都是非常虔诚的基督徒,他们愿意提前为自己找一块合适墓地. ...