浅谈单元测试

姓名:邹宇翔      学号:16019110051

转载自:

http://mp.weixin.qq.com/s/Zs1GnIKyoIJUFUeEQCRijw

【嵌牛导读】大家都知道程序编好后会经过QA的检测,同时正因为如此大家往往忽略了单元测试的重要性

【嵌牛鼻子】单元测试(UT)

【嵌牛提问】那么应该如何去进行单元测试呢?它对于自己的程序有什么好处呢?

【嵌牛正文】

什么是 UT ?

UT ( Unit Test )即单元测试。

UT 有什么价值?

大部分的开发都不喜欢写 UT,原因无非以下几点:

产品经理天天催进度,哪有时间写 UT。

UT 是测试自己的代码,自测?那要 QA 何用?

自测能测出 bug ?都是基于自身思维,就像考试做完第一遍,第二遍检查一样,基本检查不出什么东西。

UT 维护成本太高,投入产出比太低。

不会写 UT。

总之有无数种理由不想写 UT,作为工作不到三年的菜鸟深有体会。之前在点评工作的时候,团队的“UT”都集中于 RPC 的服务端。为啥带双引号?

因为 RPC 的服务端没有页面可以功能测试,部署到测试环境测试太麻烦,只能写 UT 了。在这个场景下我认为叫”验证”更合适,验证不等于测试。

验证往往只写主逻辑是否通过,且就一个 Case,且没有 Assert,有的是 System.out。

本人实习的时候做测试的,那时候知道一个测试模型。如下图:

(图一)

图的意思就是越底层做的测试效果越好,越往上则越差。也就是说大部分公司现在做的功能测试其实是效果最差的一种测试方式。

另外, QA 界有个现场:大家都知道功能测试没技术含量,那如何使自己突出呢?答案就是:自动化测试。现实是没几个公司能做好自动化测试,业界做的比较好的百度算一个。那么为啥自动化测试这么难做的?在这个模型当中,越往上黑盒越大,自动化测试难度就越大。

这句话反过来就是越往下自动化测试就越好做?没错, UT 其实是最容易实现且效果最好的自动化测试。 所以在很多公司出现一种现场: QA 写 UT。

原因总结一下就两点:开发不愿意写 UT,QA 想自动化测试解放自己。

以上的模型只是理论上说明 UT 具有巨大的价值,但是真的如此么?我只想说,只有真正尝到 UT 的好处的甜头才会意识到 UT 的价值。

Unit Test & Intergration Test

单元测试和集成测试的界线我相信大部分开发也是不清晰的。个人理解单元测试针对于一块业务逻辑最小的单元,太抽象。物理上可以简单理解为一个类的方法,可以是 public 方法也可以是 private 方法。一个单元测试不应该包含外部依赖的逻辑,反之就是集成测试了。

问题的核心就在于此。一个 service 的一个接口实现可能依赖很多第三方:1.本地其它的 service 2. dao 调用 3. rpc 调用 4.微服务调用。如下图:

(图二 )

也就是说你的单元测试,真正调用了外部依赖那就是集成测试。这其实很常见对不?我们先说这种情况下如何集成测试。

Local Integration  Test

本地集成测试也就是说不依赖与其他进程。包括: service 依赖其他本地 service 或者 dao 的情况。在讲述如何集成测试之前,我们先理一下测试模型,测试主要包含三块内容:1.数据准备 2.执行逻辑 3.输出验证。

第一步:数据准备

在本地集成测试里,数据来源基本上来自于 dao , dao 来自于 sql 。也就是在执行一个 case 之前,执行一些 sql 脚本,数据库则使用 h2 这类 memory database ,切记不要依赖公司测试环境的 db 。

下图是使用 spring - test 框架的一个 case ,可以在 case 执行之前准备我们所需要的各种数据,另外在执行完 case 之后,执行 clean.sql 脚本来清理脏数据。这里也说明一个 case 的执行环境是完全独立的, case 之间互不干扰,这很重要。

(图三)

第二步:执行逻辑最简单,就是调用一下我们测试的方法即可

第三步:验证

集成测试一般是调用 service ,或者 dao 的接口验证。

举个例子: CRUD 操作的集成测试

调用 C 接口

调用 R 接口,验证 C 成功

调用 U 接口

调用 R 接口,验证 U 成功

调用 D 接口

调用 R 接口,验证 D 成功

Remote Integration  Test

假设我们一个 service 实现依赖某个 RPC Service

第一步:数据准备

跑到别人家的数据库插几条数据?或者跟 PRC Service 的 Owner 商量好,搭一个测试环境供我们测试?有些公司还真有专门的自动化测试环境,那么即使有测试环境,那如何实现各种 case 场景下,第三方 Service 很配合的返回数据给我们?想想都蛋疼。

第二步:执行方法

假设我们成功的解决了第一步中的问题,皆大欢喜。现在来看第二步,假设我们的 service 里面调用了另一个 RPC Service 创建了很多数据,跑了无数次 case ,结果。… RPC Service 对应的数据库都是我们的脏数据,如何清理?而且他们敢随便删数据吗?想想也蛋疼。

第三步:输出验证

假设我们又愉快的解决了第二步中的问题。现在来看第三步,假设我们的方法执行最终输出是创建了一个订单,订单当然是调用订单 Service 接口了,那么我们如何验证订单是否成功创建了呢?或许可以调用订单 Service 查询订单的接口来验证。很明显大多数情况下并没有这么完美。想想也蛋疼呀。

通过以上分析, Local Integration  Test 是可行的, Remote Integration  Test 基本不可行。

那么有没有什么办法解决呢?答案就是 Mock

第一步: Mock RPC  Service 想返回什么数据就返回什么数据

第二步:还是 Mock 接口,想调用几次就调用几次

第三步:这一步等到下面讲完单元测试就明白了

Unit Test

上面我们谈到 Mock 可以解决外部依赖的问题,现在有很多 Mock 的开源框架比如: mockito。那么问题来了,既然我们可以 mock 第三方远程依赖,为何不 mock dao、local service 呢?没错外部依赖全部 mock 掉,就是单元测试了。因为我们只关心所测试的方法的业务逻辑,也就是真正高内聚的逻辑单元了。如下图:

(图四)

好处如下:

没有什么数据是造不出来的,通通返回 Mock 的对象

代码中的异常处理代码,也可以通过 mock 接口,使之抛出异常

不产生任何脏数据

跑 case 更快了,因为不用启动整个项目,相当于 Main 方法

有人会说,都 mock 了还测试个蛋蛋。

这就是对于单元测试的理解了,单元测试应该只针对于目标方法的业务逻辑测试, dao 、其它 service 应该在它们自身的单元测试去测试。对于依赖的第三方,我们应该信任它们能正确的完成我们所预期的。这句话很难理解对不对?

举几个例子

例子一:方法的最后是执行 dao 的 create 操作,那么该如何验证?

我们应该验证的内容是:

dao 的 create 方法被调用了

调用次数是对的

调用参数也是对的

没错,只要这三个验证通过,那么这个 case 执行就是通过的。因为我们相信 dao 的 create 操作能正确的完成我们所预期的,只要我们调用了正确的次数并且参数都是对的。 dao 的执行的正确性保证是在该 dao 的单元测试做的。

在 Remote Integration Test 里面第三步验证道理是一样的,我们应该验证 RPC 接口被调用了且次数和参数都是对的,那么我们的 case 就算通过了,至于, RPC 服务端是否正确执行是它们的事情不是我们所关心的。

Mockito 框架的 verify 接口就是做这件事情的。如果你理解了上述内容,那么你就开窍了, UT 不在变得这么难写。

什么时候用单元测试,什么时候用集成测试?

在本人的实践中摸索发现,对于简单的业务,比如 crud 型的瘦 service,比较适合于集成测试。

以下情况适合于单元测试:

Util 类

含有远程调用的方法

输入少,业务逻辑复杂的方法

需要异常处理的方法

case 细到什么程度为好?

这个问题也是比较经典的,一个方法要是所有的路径都覆盖到,那么要写很多的 case,说真的累死人。我的建议是两个原则:

1. 核心逻辑,容易出错的逻辑一定要覆盖到;

2. 根据自己的时间。 没必要写的非常多,毕竟 case 维护成本很高,业务逻辑一改, case 得跟着改。

总结

本人目前在从事于开源项目(Apollo(配置中心)  )研发,开源项目对代码质量要求相对来说高一些, UT 当然是很重要的一环。刚开始也不会写 UT,当然态度上也不重视 UT。

老大的代码 UT 覆盖率很高,抱着对开源负责的态度慢慢接受学习 UT ,到后来尝了几次甜头后,发现 UT 真的很实用,价值也很高,但是很遗憾 UT 被大部分开发所忽略。当然本人对 UT 的理解、实践还不够,仍需继续实践模式。

最后说一句:当开发完功能,跑完 UT,你可以放心的上线了的时候,你的 UT 就成功了。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容

  • 本文作者:张乐。全文约 4199 字,读完可能需要 7 分钟。虽然这篇不是以 Python 为示例的,但基本的思路...
    罗义的夏天阅读 917评论 0 3
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • Android单元测试介绍 处于高速迭代开发中的Android项目往往需要除黑盒测试外更加可靠的质量保障,这正是单...
    东经315度阅读 3,100评论 6 37
  • 那场雨从未停过,落在那年,落在小城,烙下印记在我心间。 我空有满腹诗文,却难以表达我的心。 “小生寒窗苦读十余载,...
    柯望一阅读 9,577评论 3 8
  • 01 十月二十二日晚上听了笑笑梨老师的课,让我受益匪浅啊!虽然教室很小,但是我感受到了课堂的氛围。来上课的同学...
    写作星阅读 636评论 3 3