概述

由于最近工作需要, 在项目中要做单元测试, 以达到指定的测试用例覆盖率指标。项目中我们引入的powermockito来编写测试用例, JaCoCo来监控单元测试覆盖率。关于框架的选择, 网上讨论mockito和powermockito孰优孰劣的文章众多, 这里就不多做阐述, 读者如有兴趣可自行了解。

依赖引入

<dependency>
<groupId>org.powermock</groupId>
<artifactId>powermock-module-junit4-rule-agent</artifactId>
<version>1.6.6</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.powermock</groupId>
<artifactId>powermock-api-mockito</artifactId>
<version>1.6.6</version>
</dependency>
<dependency>
<groupId>org.powermock</groupId>
<artifactId>powermock-module-junit4</artifactId>
<version>1.6.6</version>
</dependency>

被测试类

public class PowerMockitoDemo {

    @Autowired
private StudentDao studentDao; @Autowired
private TeacherService teacherService; public void study() {
//doSomething
} private void play(Map<String, Object> project, Person person, int hours) {
//doSomething
} private boolean updateStudentName(String newName) {
//doSomething
} public ServiceResult grantRights(List<String> usernames, String rights, String orderid, String id) {
String value1 = PropertiesUtil.get("key1");
String value2 = PropertiesUtil.get("key2");
studentDao.saveRecord(usernames);//返回值类型void
Student student = studentDao.getStudentById(id);
boolean result = this.verifyParams(usernames);
teacherService.syncDB2Redis(usernames, orderid);
this.updateOperation(rights);
} private boolean verifyParams(List<String> usernames) {
//doSomething
} private void updateOperation(String rights) {
//doSomething
}
}

测试用例基类

//@PrepareForTest注解和@RunWith注解需结合使用,单独使用将不起作用
@RunWith(PowerMockRunner.class)
@PrepareForTest({RedisUtils.class})
@SuppressStaticInitializationFor({"com.test.util.RedisUtils", "com.test.util.HttpUtils"})//用于阻止类中的静态代码块执行
public abstract class BaseTest { public RedisUtils redisUtils; @Rule
public ExpectedException thrown = ExpectedException.none();//断言要抛出的异常 public void setUp() {
initMocks(this); PowerMockito.suppress(PowerMockito.constructor(ShardedJedisClientImpl.class, String.class));
redisUtils = PowerMockito.mock(RedisUtils.class);
Whitebox.setInternalState(RedisUtils.class, "redisUtil", redisUtils);//给类或实例对象的成员变量设置模拟值,这里是给RedisUtils类中的字段redisUtil设置模拟值 PowerMockito.suppress(PowerMockito.constructor(HttpUtils.class));
PowerMockito.mockStatic(HttpUtils.class);//mock类中所有静态方法
} /**
* @param instance 真实对象
* @param methodName 方法名
* @param args 形参列表
*/
public Object callPrivateMethod(Object instance, String methodName, Object... args) throws Exception {
return Whitebox.invokeMethod(instance, methodName, args);//调用私有方法
} }

测试用例

@PrepareForTest({PowerMockitoDemo.class, PropertiesUtil.class})//此处PowerMockitoDemo被测试类添加到@PrepareForTest注解中, 用于mock其静态、final修饰及私有方法;另外,PropertiesUtil工具类由于不通用,不适合抽取到基类BaseTest中, 可在子类mock
@SuppressStaticInitializationFor("com.test.util.PropertiesUtil")//用于阻止类中的静态代码块执行
public class PowerMockitoDemoTest extends BaseTest{ @org.powermock.core.classloader.annotations.Mock
private StudentDao studentDao; @org.powermock.core.classloader.annotations.Mock
private TeacherService teacherService; @org.powermock.core.classloader.annotations.Mock
@InjectMocks
private PowerMockitoDemo powerMockitoDemo; @Override
@Before
public void setUp() {
super.setUp();
PowerMockito.suppress(PowerMockito.constructor(PropertiesUtil.class));
PowerMockito.mockStatic(PropertiesUtil.class);//mock类中所有静态方法
} @Test
public void studyWhenCallSuccessfully() {
PowerMockito.doCallRealMethod().when(powerMockitoDemo).study();
//doSomething powerMockitoDemo.study();
} @Test
public void playWhenCallSuccessfully() {
Map<String, Object> project = new HashMap<String, Object>();
Person person = new Person();
int hours = 8; PowerMockito.doCallRealMethod().when(powerMockitoDemo, "play", Matchers.anyMapOf(String.class, Object.class), Matchers.any(Person.class), Matchers.anyInt());
//doSomething this.callPrivateMethod(powerMockitoDemo, "play", project, person, hours);
} @Test
public void updateStudentNameWhenCallSuccessfully() throws Exception {
String id = "9527"; PowerMockito.when(powerMockitoDemo, "updateStudentName", Matchers.anyString()).thenCallRealMethod();
//doSomething boolean actualResult = this.callPrivateMethod(powerMockitoDemo, "updateStudentName", id);
Assert.assertTrue(actualResult == true);
} @Test
public void getStudentByIdWhenCallSuccessfully() throws Exception {
List<String> usernames = new ArrayList<String>();
String rights = "万叶飞花流";
String orderid = "orderid";
String id = "id"; //调用真实方法
PowerMockito.when(powerMockitoDemo.grantRights(Matchers.anyListOf(String.class), Matchers.anyString(), Matchers.anyString(), Matchers.anyString())).thenCallRealMethod();
//当方法内重复调用同一个方法时, 可通过Matchers.eq()方法来指定实际入参来加以区分
PowerMockito.when(PropertiesUtil.get(Matchers.eq("key1"))).thenReturn("value1");
PowerMockito.when(PropertiesUtil.get(Matchers.eq("key2"))).thenReturn("value2");
//返回值类型为void,不做任何事情
PowerMockito.doNothing().when(studentDao).saveRecord(Matchers.anyListOf(String.class));
//类似需要调用数据库、redis、远程服务的,可直接模拟返回值,不做方法的真实调用
PowerMockito.when(studentDao.getStudentById(Matchers.anyString())).thenReturn(new Student());
//调用真实的私有方法
PowerMockito.when(powerMockitoDemo, "verifyParams", Matchers.anyListOf(String.class)).thenCallRealMethod();
//模拟私有方法返回值
PowerMockito.when(powerMockitoDemo, "verifyParams", Matchers.anyListOf(String.class)).thenReturn(true);
//模拟方法调用抛出异常。当被调用的方法头没有显式声明异常时, 则mock只支持unchecked exception,比如这里syncDB2Redis()方法签名没有声明任何异常,则thenThrow()模拟异常时只支持模拟运行时异常,使用非运行时异常将编译不通过
PowerMockito.when(teacherService.syncDB2Redis(Matchers.anyListOf(String.class), Matchers.anyString())).thenThrow(new RuntimeException("Failed to call remote service"));
PowerMockito.doNothing().when(powerMockitoDemo, "updateOperation", Matchers.anyString()); ServiceResult expectedResult = new ServiceResult();
ServiceResult actualResult = powerMockitoDemo.grantRights(usernames, rights, orderid, id);
//断言实际调用结果是否符合预期值
Assert.assertEquals(JSON.toJSONString(expectedResult), JSON.toJSONString(actualResult));
}
}

如上, 具体的阐释在代码注释中都已经标注, 抽取基类是为了提高代码可重用性。博主这里是为了演示, 所以代码看起来会有点臃肿, 在实际项目使用中, 可以通过静态引入 import static org.mockito.Mockito.when; 和 import static org.mockito.Matchers.*; 来简化代码, 提高可阅读性。

另外由于被测试类在测试方法中被mock掉, 且被@PrepareForTest注解标记时, JaCoCo工具统计测试覆盖率将忽略该测试类。可通过在基类BaseTest中添加 @Rule public ExpectedException thrown = ExpectedException.none(); , 并去掉 @RunWith(PowerMockRunner.class) 和 @SuppressStaticInitializationFor 来使统计测试覆盖率生效。但这里又有一个新问题产生, 基类修改之后, 发现测试用例无法进入debug调试, 因此建议先用修改前的基类来编写单元测试, 便于调试, 待测试用例完成后, 再修改基类令Jacoco统计覆盖率生效。

关于PowerMockito的实践, 博主目前在项目中的使用主要就涉及到了这些, 所以这次做了回标题党"XXX深入实践" ^_^, 后面如有接触新的相关知识点, 会陆续更新到本篇文章中。如有错误, 欢迎指正, 谢谢你^_^

参考资料

非web下的PowerMockito单元测试

powermockito单元测试之深入实践的更多相关文章

  1. [转载]单元测试之道(使用NUnit)

    首先来看下面几个场景你是否熟悉 1.你正在开发一个系统,你不断地编码-编译-调试-编码-编译-调试……终于,你负责的功能模块从上到下全部完成且编译通过!你长出一口气,怀着激动而又忐忑的心情点击界面上的 ...

  2. 单元测试之道(使用NUnit)

    首先来看下面几个场景你是否熟悉 1.你正在开发一个系统,你不断地编码-编译-调试-编码-编译-调试……终于,你负责的功能模块从上到下全部完成且编译通过!你长出一口气,怀着激动而 又忐忑的心情点击界面上 ...

  3. 补习系列(8)-springboot 单元测试之道

    目录 目标 一.About 单元测试 二.About Junit 三.SpringBoot-单元测试 项目依赖 测试样例 四.Mock测试 五.最后 目标 了解 单元测试的背景 了解如何 利用 spr ...

  4. ASP.NET Core搭建多层网站架构【3-xUnit单元测试之简单方法测试】

    2020/01/28, ASP.NET Core 3.1, VS2019, xUnit 2.4.0 摘要:基于ASP.NET Core 3.1 WebApi搭建后端多层网站架构[3-xUnit单元测试 ...

  5. iOS 单元测试之XCTest详解(一)

    iOS 单元测试之XCTest详解(一) http://blog.csdn.net/hello_hwc/article/details/46671053 原创blog,转载请注明出处 blog.csd ...

  6. 玩转单元测试之Testing Spring MVC Controllers

    玩转单元测试之 Testing Spring MVC Controllers 转载注明出处:http://www.cnblogs.com/wade-xu/p/4311657.html The Spri ...

  7. 玩转单元测试之WireMock -- Web服务模拟器

    玩转单元测试之WireMock -- Web服务模拟器 WireMock 是一个灵活的库用于 Web 服务测试,和其他测试工具不同的是,WireMock 创建一个实际的 HTTP服务器来运行你的 We ...

  8. 单元测试之NSNull 检测

    本文主要讲 单元测试之NSNull 检测,在现实开发中,我们最烦的往往就是服务端返回的数据中隐藏着NSNull的数据,一般我们的做法是通过[data isKindOfClass:[NSNull cla ...

  9. 使用VisualStudio进行单元测试之二

    借着工作忙的借口,偷了两天懒,今天继续单元测试之旅.前面说了如何进行一个最简单的单元测试,这次呢就跟大家一起来熟悉一下,在visual studio中如何进行数据驱动的单元测试. 开始之前先来明确一下 ...

随机推荐

  1. hdoj2037 贪心算法——今年暑假不AC

    所谓“贪心算法”是指:在对问题求解时,总是作出在当前看来是最好的选择.也就是说,不从整体上加以考虑,它所作出的仅仅是在某种意义上的局部最优解(是否是全局最优,需要证明). 经典问题:时间序列问题   ...

  2. 给定一个整数数组和一个目标值,找出数组中和为目标值的两个数 例如给定nums = [2,7,11,15],target = 9

    python解决方案 nums = [1,2,3,4,5,6] #假如这是给定的数组 target = 9 #假如这是给定的目标值 num_list = [] #用来装结果的容器 def run(nu ...

  3. 1. Django每日一码 之原生View源码

    2019-7-4  今日源码:原生 View 1.URL中调用as_view方法 def as_view(cls, **initkwargs): """ Main ent ...

  4. Spring Boot 邮件发送的 5 种姿势!

    邮件发送其实是一个非常常见的需求,用户注册,找回密码等地方,都会用到,使用 JavaSE 代码发送邮件,步骤还是挺繁琐的,Spring Boot 中对于邮件发送,提供了相关的自动化配置类,使得邮件发送 ...

  5. [Spring+SpringMVC+Mybatis]框架学习笔记(六):事务

    第7讲 事务 7.1 事务的概念 事务是一系列作为一个逻辑单元来执行的操作集合. 它是数据库维护数据一致性的单位,它讲数据库从一个一致状态,转变为新的另外一个一致状态.说的简单一点就是:如果一组处理步 ...

  6. Linux磁盘与分区

    正在从新装载虚拟机,碰到磁盘分区一阵头大,花了一下午对分区的基本原理做了一个梳理   一.磁盘   硬盘内部结构:

  7. 【小家Spring】聊聊Spring中的数据绑定 --- DataBinder本尊(源码分析)

    每篇一句 唯有热爱和坚持,才能让你在程序人生中屹立不倒,切忌跟风什么语言或就学什么去~ 相关阅读 [小家Spring]聊聊Spring中的数据绑定 --- 属性访问器PropertyAccessor和 ...

  8. 判断list集合不为空

    在java开发中新手容易将判断一个list集合是否为空,只以If(list!=null)去判断,且容易和isEmpty()混淆,但是,list集合为空还是为null,是有区别的. 先看一下下面的例子, ...

  9. HBase部署与使用

    HBase部署与使用 概述 HBase的角色 HMaster 功能: 监控RegionServer 处理RegionServer故障转移 处理元数据的变更 处理region的分配或移除 在空闲时间进行 ...

  10. C#各版本新增加功能

    本系列文章主要整理并介绍 C# 各版本的新增功能. C# 8.0 C#8.0 于 2019年4月 随 .NET Framework 4.8 与 Visual Studio 2019 一同发布,但是当前 ...