单元测试中mock的使用及mock神器jmockit实践

在最近的r应用的单元测试中,经常需要用到mock,可以说mock在ut (unit test)中是无处不在的。而在r的ut实践中也找到了一种很简洁的mock方式,不仅解决了ut中所有需要mock的地方,而且可以很少量的代码来完成mock。详见下文。

  一.Mock的使用场景:

  比如以下场景:

  1. mock掉外部依赖的应用的HSF service的调用,比如uic,tp 的hsf服务依赖。

  2. 对DAO层(访问mysql、oracle、tair、tfs等底层存储)的调用mock等。

  3. 对系统间异步交互notify消息的mock。

  4. 对method_A里面调用到的method_B 的mock 。

  5. 对一些应用里面自己的 class(abstract, final, static),interface,annotation ,enum,native等的mock。

  二. Mock工具的原理:

  mock工具工作的原理大都如下:

  1. record阶段:录制期望。也可以理解为数据准备阶段。创建依赖的class 或interface或method ,模拟返回的数据,及调用的次数等。

  2. replay阶段:通过调用被测代码,执行测试。期间会invoke 到 第一阶段record的mock对象或方法。

  3. verify阶段:验证。可以验证调用返回是否正确。及mock的方法调用次数,顺序等。

  三. 当前的一些Mock工具的比较:

  历史曾经或当前比较流行的Mock工具有EasyMock、jMock、Mockito、Unitils Mock、PowerMock、jmockit等工具。

  他们的功能对比如下:

  从这里可以看到,当前为什么jmockit为什么这么火爆了!所以我们的UT中的mock工具也选择了目前无所不能的jmockit。

  而在使用的过程中,感觉到jmockit的 Auto-injection of mocks 及 Special fields for "any" argument matching  及各种有用的 Annotation 给测试代码精简和测试效率提升带来了实实在在的好处。



  四. Jmockit的简介:

  JMockit 是用以帮助开发人员编写测试程序的一组工具和API,它(https://code.google.com/p/jmockit/)完全基于 Java 5 SE 的 java.lang.instrument 包开发,内部使用 ASM 库来修改Java的Bytecode。正是由于基于instrument,可以修改字节码。所以这也是它强大的原因。

  Jmockit可以mock的种类包含了:1.class(abstract, final, static) ;2.interface ;3.enum ;4.annotation ;5.native 。

  Jmockit 有两种mock的方式:

  1. Behavior-oriented(Expectations & Verifications)  ;

  2. State-oriented(MockUp<GenericType>)   。

  通俗点讲,Behavior-oriented是基于行为的mock,对mock目标代码的行为进行模仿,更像黑盒测试。State-oriented 是基于状态的mock,是站在目标测试代码内部的。可以对传入的参数进行检查、匹配,才返回某些结果,类似白盒。而State-oriented的 new MockUp基本上可以mock任何代码或逻辑。非常强大。

  以下是jmockit的APIs和tools:

  可以看到jmockit常用的Expectation、NonStrictExpectations 期望录制 及Annotation @Tested、@Mocked,@NonStrict、@Injectable 等简洁的mock代码风格。而且jmockit 还自带了code coverage的工具供本地UT时候看逻辑覆盖或代码覆盖率使用。

  五.Jmockit的实践:

  第一步:添加jmockit的jar包依赖

  在refund的单元测试过程中,第一步:应用pom中引入jmockit的jar包,可以顺带引入jmockit自带的code coverage的jar。

  第二步:一个完整的Jmockit的示例:


  这个是对refundmanager里面查询可退款金额范围 queryRefundFeeRange的单元测试。通过看被测代码可以看到这个方法的实现里面调用了

  feeResultDO = confirmGoodsService.queryConfirmToSellerRefundFee(detailId);

  这个外部的hsf依赖 获取了feeResultDO 。

  这个hsf调用是需要mock的。

  传统的mock或ut ,对 confirmGoodsService 这个bean是需要初始化,通过spring的配置,初始化加载等 一大堆代码。

  jmockit通过了注解的方式:

@Injectable
private ConfirmGoodsService confirmGoodsService ;

  一个Annotation就搞定了所有的配置,加载等问题。直接复用开发代码里面的bean,节省了大量的代码。

  另外  @Tested  RefundManagerImpl refundManagerImpl = new RefundManagerImpl();

  这里也用到了注解 @Tested   表示被测试的class 。

  另外还有常用的注解:@Mocked,@NonStrict等。

  而这段代码就是mock的核心:录制被mock的method的行为及期望返回:

new Expectations(){
{
confirmGoodsService.queryConfirmToSellerRefundFee(anyLong);
result = feeResultDOmock;
times = 1;
}
} ;

  其中

  result  可以返回任意需要的测试类型;

  times  表示期望被调用的次数。

  是不是看起来非常简洁明了。

  而上面该段代码如果 换成基于状态的mockup 代码如下:

  采用MockUp的方式,可以mock任意的mock对象或方法,因为它直接改写了原method的实现逻辑,直接返回需要的数据。

  这也是jmockit彪悍的地方之一。

  最后数据回收,防止各个testcase的mock相互影响的方式:

  Mockit.tearDownMocks();

  这一步也可以省略。

  还要重点介绍的就是mock期望里面的入参 any。


这个any系列的万能入参类型,也可以节省很多mock代码,可以高效的准备任何入参类型。


  以上,一个最简单的,也最实用的jmockit的示例。


  jmockit的更多,对interface及method的单元测试的示例,将在后续总结汇总。


  六、 Jmockit自带的code coverage :


  工程的 pom文件中引入 jmockit-coverage 后,本地eclipse启动单元测试后, 会自动统计单元测试的代码覆盖率。关于行覆盖,方法覆盖,类覆盖,分支逻辑覆盖等各种数据都可以看到。


  IDE启动UT时候,加载 code coverage  组件,


  点击进去,可以看到具体的覆盖逻辑:


  其中绿色部分表示源代码被run过。


  代码覆盖对指导单元测试的测试逻辑,覆盖等提供了直观的指示。


  以上,就是在单元测试中mock技术的应用:Jmockit的使用介绍及实际应用示例。它在单元测试中确实可以很少的代码mock掉外部依赖,提高ut的效率,并且 自带的code coverage可很方便的看到ut对被测代码的覆盖效果,指导测试设计。


  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值