再谈Mock Object

发现静寂的夜能让我的思维加快。

Mock Object进行Unit Test已经一周多了,发现以前对Mock Object还是很肤浅,即使是现在我也不敢说我对Mock Object的理解就一定正确。

这篇blog假设你已经熟悉JUnit、了解Mock和TDD
如果你是直接开始使用JMock 、Easy Mock或者是MockMaker等Mock Object框架的,我建议你简单了解一下Mock Object的演化历史,这样你在使用Mock Object时才会更有的放矢。

Mock object有关键的两个概念:
* 建立起环境的概念。在www.mockobjects.com的faq中有一句是这样叙述的“I think the fundamental thing to remember about Mock objects is that they are just that – simple shells or placeholders.” Mock object只是替代了被测Object环境的代码。
* test assertions被隐藏在Mock object内部实现中了,在你的test case中用来verify被测代码与Mock object的交互。

在单元测试中,人们发现有一些问题(这些问题在我的“认识Mock Object”中已列出)常见单元测试工具(如JUnit)并不能很好、很便捷的解决,这样Mock object被引入来解决这些问题。

最初人们手工编写Mock Object。随着测试问题越来越复杂,人们自己手工编写的Mock object越来越多,人们开始将编写Mock Object过程中一些通用的东西抽象出来,形成了一些Mock Object Lib,以帮助开发人员快速得到自己需要的Mock Objects。

当前的Mock Object Lib有多种,大致可分为两类:
* Static Mock Objects Lib
* Dynamic Mock Objects Lib

这里简单举例说明一下不同类型Mock object lib的使用方法,并与手工编写进行对比。

问题:我们在测试某个class时,我们需要与MyInterface这个接口进行交互,而该接口尚未实现,这时我们使用Mock object来替代。

接口MyInterface:
public interface MyInterface {
    public SomeClass getSomething();
    public void setSomething(SomeClass aSomething)
        // Other methods omitted…
}

* 手工编写mock object:
public class HandcraftMockMyInterface implements MyInterface {
    public SomeClass getSomething() {
        return something;
    }
    public void setSomething(SomeClass aSomething){
        this.someting = aSomething;
    }
    private SomeClass something;
    //others
}

* 使用Static Mock Object Lib编写Mock Interface:
public class StaticMockMyInterface extends MockObject implements MyInterface{
    private final ExpectationValue something = new ExpectationValue("something");  
    public void setSomething(SomeClass something){
        this.something.setActual(something);
    }
    public void setExpectedSomething(SomeClass something){
        this.something.setExpected(something);
    }
    public void SomeClass getSomething(){
        return this. something;
    }  
}

* 使用Dynamic Mock Object Lib编写Mock Interface:(以JMock为例)
Mock dynamicMockMyInterface = new Mock(MyInterface.class);
MyInterface mi = (MyInterface) dynamicMockMyInterface.proxy();

SomeClass someThing = new SomeClass();
dynamicMockMyInterface.expects(once()).method(“setSomething”).with(eq(someThing));
dynamicMockMyInterface.expects(once()).method(“getSomething”).will(returnValue(someThing));

//执行你的测试代码,比如你的tested object与MyInterface mi交互的代码。

dynamicMockMyInterface.verify();

Mock Object的使用流程
- Setup any state — setup the fixture for your test
- Set expectations for the test
- Run the target code
- Verify that your expectations have been met

Mock Object Practical Experience
- 好的设计是容易测试的设计
- 尽量让测试在内存中完成,不要在硬盘上留下“垃圾”
- 面向接口使用Mock Object。better to implement interface , not inherit class

写在工作时

不知怎么的,我总是喜欢在工作的时候来完成我的blog,这也就是我这篇blog的题目来由。

享受完美妙的平安夜和快乐的圣诞节,又开始我新一天的工作。

北国寒冬,从寝室出来,道路上几乎不见行人,身上的热气不断逃离,寒气扑来,打了几个寒颤。心里想着今天的工作,脚下步伐却在不知不觉中加快。

来到温暖的办公区,打开outlook,看到一份来自远方好朋友的邮件,邮件中如是说“你的blog也不更新。看到梦想风暴的圣诞前夜写的blog,感慨挺多的。”看完后的第一感觉是很兴奋,因为第一次有人提醒我该更新我的blog了;但是心里也不免有些惭愧,毕竟看到我的最后一篇blog是12月15号的,掐指算来已有近半个月没有动笔了,想了想这段时间我的“所作所为”,除了公司项目的更迭,再就是Dominoo的tdd(test driven development)。从大学接触计算机程序以来,一直是直接完成功能代码然后测试的,突然让我先写test code然后完成business code,我的思维很难做出这个“急转弯”,心情也是很郁闷的,时间在不停的流逝,而代码却被反复的推翻。dreamhead看出了这点,决定和我结对编程,一步步带我形成测试驱动开发的思维。经过那次结对后发现自己正向着tdd靠拢,加上昨天一天自己的亲自试验,发现效果还是不错的。这周是2004年的最后一周,对我们的dominoo来说也是很重要的一周,因为我们的计划是这周拿出一个“初具规模”的内部版本,我们现在都在为此而努力着。

自从dreamhead向我推荐blog以来,我逐渐的开始喜欢阅读其他人的blog了。打开dreamhead的“剧本集”(blog),也欣赏一下那篇well-known的“写在平安夜”,看看dreamhead发的“牢骚”^_^,居然看到他的“剧本”中还有我的“角色”^_^。

由于一个好朋友的内部推荐,昨天接到了国内某著名大公司的面试邀请,不过被我宛然谢绝了,我刚刚入司不到1年,虽说这里并不像我想象中的那么有激情而且待遇又不是令我满意,但是能遇到像dreamhead、darwin_yuan、model和toidi_xu这样的对软件如此痴迷的而又满怀激情的人,我还是蛮欣慰的。我们成为朋友是因为我们志同道合,我们有自己的共同的梦想,我们有着同样的激情。我们还要在一起共同的奋斗,我暂时还离不开这些朋友。

在写这篇blog时又看到了远方那位好友回复的邮件,说他正在准备一篇要投到一个国际会议并且EI检索的paper,我知道这是他的第一篇正式要发表的paper,真心祝愿他能成功。

如发现本站页面被黑,比如:挂载广告、挖矿等恶意代码,请朋友们及时联系我。十分感谢! Go语言第一课 Go语言精进之路1 Go语言精进之路2 Go语言编程指南
商务合作请联系bigwhite.cn AT aliyun.com

欢迎使用邮件订阅我的博客

输入邮箱订阅本站,只要有新文章发布,就会第一时间发送邮件通知你哦!

这里是 Tony Bai的个人Blog,欢迎访问、订阅和留言! 订阅Feed请点击上面图片

如果您觉得这里的文章对您有帮助,请扫描上方二维码进行捐赠 ,加油后的Tony Bai将会为您呈现更多精彩的文章,谢谢!

如果您希望通过微信捐赠,请用微信客户端扫描下方赞赏码:

如果您希望通过比特币或以太币捐赠,可以扫描下方二维码:

比特币:

以太币:

如果您喜欢通过微信浏览本站内容,可以扫描下方二维码,订阅本站官方微信订阅号“iamtonybai”;点击二维码,可直达本人官方微博主页^_^:
本站Powered by Digital Ocean VPS。
选择Digital Ocean VPS主机,即可获得10美元现金充值,可 免费使用两个月哟! 著名主机提供商Linode 10$优惠码:linode10,在 这里注册即可免费获 得。阿里云推荐码: 1WFZ0V立享9折!


View Tony Bai's profile on LinkedIn
DigitalOcean Referral Badge

文章

评论

  • 正在加载...

分类

标签

归档



View My Stats