前端测试常见的 3 个误区

前言

哈喽,大家好,我是海怪。

在做前端测试时,选用合适的测试策略远比一通猛狂测试更重要,所谓 “方向 > 努力”

如果选择了错误的测试策略,很容易写出维护性差和不稳定的测试用例。一旦业务出现变化,用例就全崩了。可能这也是大家讨厌写测试的原因之一吧。

Kent C. Dodds 在这篇文章 《Common Testing Mistakes
分享 了 3 个他看到的测试误区。今天,就把这篇文章分享给大家吧~

翻译中会尽量用更地道的语言,这也意味着会给原文加一层 Buf,想看原文的可点击 这里


正片开始

误区一:测试代码的实现细节

说实话,我非常喜欢这个误区(详情可以看这里),因为在测试过程中,它是一个很严重的问题,这样写测试也不会带给你对应的信心。下面是一个测试代码实现细节的例子:

// counter.js
import * as React from 'react'

export class Counter extends React.Component {
  state = {count: 0}
  increment = () => this.setState(({count}) => ({count: count + 1}))
  render() {
    const {count} = this.state
    return <button onClick={this.increment}>{count}</button>
  }
}

// __tests__/counter.js
import * as React from 'react'

// 用 React Testing Library 是很难测代码实现细节的,所以这里用 enzyme 来测
import {mount} from 'enzyme'
import {Counter} from '../counter'

test('the increment method increments count', () => {
  const wrapper = mount(<Counter />)
  // 千万别这么做
  expect(wrapper.instance().state.count).toBe(0)
  wrapper.instance().increment()
  expect(wrapper.instance().state.count).toBe(1)
})

为什么说上面是在测代码实现细节?以及,为什么测代码细节是不好的呢?像上面那样过度测试实现细节会带来两个结果:

  • 我可以在测试完全通过的情况下弄崩业务代码(比如在 onClick 赋值时故意写错变量名)
  • 我可以在重构业务代码的时候弄崩测试用例(例如,把 increment 重命名为 updateCount,测试就崩了,但业务代码是能正常运行的)

(译注:作者对重构的理解是:改动业务代码逻辑时,测试代码不应该做改动的,因为业务逻辑没变,只是实现方式变了

类似这样的测试用例是最难维护的,因为你要不断地更新它们(基于上面第二点),同时,它们也不会给你带来更多代码的信心(基于上面第一点)。

误区二:100% 代码覆盖率

另一个误区就是强求 100% 的代码覆盖率。 有趣的是,我经常看到在一些项目里会被强制要求 100% 的覆盖率。不管这种规定是从哪来的,这其实都是对代码覆盖的误解,因为这样并不能给你带来相与之对应的代码信心。

代码覆盖只能告诉你一件事:

  • 这行代码有被测试用例跑过

然而,它没有告诉你的事有:

  • 代码是否按业务需求来正常工作
  • 代码是否能和项目里其它代码一起工作
  • 项目崩了的时候会发生什么(这里指意外崩溃)

代码覆盖率的另一个问题是:每增加一行代码的覆盖,整体覆盖量也会被增加。也就是说,如果你想提高整体覆盖率,那么在 “支付页” 添加测试和在 “关于” 页添加测试的效果是一样的(整体覆盖率变高了)。比这更严重的另一个问题是:这样的覆盖率不能让你深入地了解你的项目...

目前来说,还没有一种万能的解决方案来获得准确的代码覆盖率,毕竟每个项目的需求是不同的。我一般不会过度关注代码覆盖率,而是更关注于项目里重要的部分是否覆盖到位。在确定项目中的关键部分之后,我会利用覆盖率报告来找出还未被测试覆盖到的边界情况。

声明一下,对于开源模块来说,100% 代码覆盖率是完全合理的,因为它们一般更容易达到 100% 覆盖率(项目不大,而且相对简单),而且他们都是很重要的代码,会被很多别的项目引用。

误区三:重复测试

相比于集成测试和单测,大多数人吐槽 E2E 最多就是很慢和不可靠。 你是不可能让单个 E2E 测试既能跑得快,又能像单测那样稳定的。反正就是不可能的。不过话说回来,单个 E2E 测试会比单测带来更多代码信心。在很多情况下,单测是不能像 E2E 那样带来那么高的代码信心的,所以项目中写点 E2E 测试是肯定值回本的!

当然,上面这么说不代表我们不能让我们的 E2E 测试跑更快和变得更可靠。其中,重复测试是人们写 E2E 测试时经常踩的一个坑,这会让降低整个测试的性能以及可靠性。

我们应该要在隔离环境下执行测试。 理论上,每个单独的 E2E 测试在执行时都应该像不同的用户使用软件一样。这样的话,每次跑测试都要走一遍注册登录流程来创建新用户了,对吧?看起来好像是对的,然后你每次就要点点按钮,输入用户信息来做注册登录。这么做只是为了业务中要用一下用户的登录态,是吧?错!这是不对的!

让我们回过头来想:为什么要写测试?因为这样你可以交付出更有自信、不容易崩溃的项目呀!假如,你有 100 个测试需要用已登录的用户来执行。那你应该要跑多少次注册/登录流程来让你相信代码是没问题呢?100 次还是 1 次?正常人都会选 1 次,因为只要 1 次成功,处处都应该成功。因此,剩下的 99 次额外的测试并不会给你带来任何代码信心。它们只是在做无用功。

那你应该怎么做呢?既然我们已经搭建好了测试的隔离环境,那么就不应该在测试之间共享同一个 user我推荐的做法是:当每次要注册和登录新用户时,在项目中发送同一个 HTTP 请求!发送请求肯定比在页面点击选中输入框和输入用户名、密码来得更快,而且会产生更少的假错误 (译注:假错误是指:测试失败了,但是其实应用代码本身没任何问题) 。只要你能保证有 1 个能完整跑通注册/登录的流程,那么你就不会失去这个登录注册流程的信心。

总结

一定要时刻记住我们写测试是为了提高代码信心。如果你现在做的事不能让你提高对代码的信心,那可以考虑你是否真的要这么做!


好了,这篇外文就给大家带到这里了。这篇文章主要列举了 3 个误区:避免过度测试代码细节、避免 100% 覆盖以及避免重复测试。这三个误区的产生都是因为我们没有搞清楚测试的本质:提高代码自信。当你很痛苦地编写测试用例的时候,那么很可能你钻入了牛角尖,往错误的方向写测试了,这时就要停止然后回过头来想:怎么做才能提高代码自信呢?

如果你喜欢我的分享,可以来一波一键三连,点赞、在看就是我最大的动力,比心 ❤️

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

推荐阅读更多精彩内容

  • 大家好,我是十一。 前情回顾 上篇我们讲了常见测试误区,我们先来回顾下: 常见测试误区1.测试和开发是对头 正确打...
    觅识堂的十一阅读 247评论 0 0
  • 开篇词 | 从“小工”到“专家”,我的软件测试修炼之道 随着自动化测试用例设计与开发、测试框架选型、测试框架自行研...
    成功在于实践阅读 1,133评论 0 4
  • 开篇词 | 从“小工”到“专家”,我的软件测试修炼之道 随着自动化测试用例设计与开发、测试框架选型、测试框架自行研...
    成功在于实践阅读 741评论 0 1
  • 为什么要进行测试 测试可以确保得到预期的结果 作为现有代码⾏为的描述 促使开发者写可测试的代码,⼀般可测试的代码可...
    强某某阅读 919评论 0 3
  • 过于关注实现细节的测试 在为前端项目编写测试用例的时候,你也许和我一样,曾遇到过以下困扰: 明明进行了功能正确的改...
    ThoughtWorks阅读 221评论 0 0