编写高质量的测试
今天golang学习网给大家带来了《编写高质量的测试》,其中涉及到的知识点包括等等,无论你是小白还是老手,都适合看一看哦~有好的建议也欢迎大家在评论留言,若是看完有所收获,也希望大家能多多点赞支持呀!一起加油学习~

不幸的是,测试在许多组织中仍然没有得到应有的关注。有时,如果开发人员没有编写任何测试,他们会感到内疚,同时测试代码往往没有得到适当的审查。相反,评论中经常检查的唯一事情是是否有任何测试,这是一种耻辱,因为仅仅进行测试还不够好。实际上,它们至少应该与项目中的所有其他代码具有相同的质量,即使不是更高的质量。否则,测试确实可能会阻碍你,因为测试失败的次数太多,难以理解,或者运行时间太长。我已经在关于使用内存中实现而不是存储库模拟的博客文章中讨论了其中的一些要点。现在我想讨论一些其他的、更一般的、我在编写测试时要注意的事情。
极简主义是关键
stack overflow 要求您为问题添加最少的、可重现的示例,在我看来,这对于出于完全相同的原因编写测试也是非常好的建议。特别是在编写测试几个月后阅读测试时,如果发生的事情较少,就更容易完全理解正在发生的事情。因此只编写测试绝对必要的代码,并抵制仅仅因为这样做很容易就添加更多内容的诱惑。但测试代码当然仍然必须完整,即测试应包含尽可能多的行,但尽可能少。
追求 100% 的代码覆盖率
这可能是一个不受欢迎的观点,但我认为以 100% 代码覆盖率为目标是完全有意义的,尽管许多人似乎认为这是一种不好的做法。
有时团队会选择较低的值,例如代码覆盖率达到 90%。然而,这对我来说没有多大意义。首先,所有这些数字都有些随意,并且很难使用数据进行备份。此外,在编写新代码时,并非所有代码都需要经过测试才能通过该阈值。如果有人设法提高覆盖率,那么下一个人可能根本不编写任何测试,同时仍然保持高于 90% 的代码覆盖率,这会导致错误的自信感。
我经常听到的借口之一是为 getter 和 setter 等简单函数编写测试没有意义。也许令人惊讶的是,我完全同意这一点。但这里有一个问题:如果没有一个测试真正使用这些 getter 和 setter,那么可能就没有必要使用它们。因此,与其抱怨实现 100% 测试覆盖率有多么困难,最好不要首先编写不需要的代码。这也避免了每行代码带来的维护负担。
但是,有一个小问题:有时代码会执行奇怪的操作,这可能会导致代码覆盖工具将某些行标记为未覆盖,即使它是在测试运行期间执行的。我没有经常遇到这样的情况,但如果没有办法使这项工作正常进行,我会将它们排除在代码覆盖范围之外。例如。 phpunit 允许使用他们的 codecoverageignore 注释来做到这一点:
<?php
class someclass
{
/**
* @codecoverageignore
*/
public function dosomethingnotdetectedascovered()
{
}
}
这样这个函数就不会被包含在代码覆盖率分析中,这意味着仍然有可能达到 100% 的代码覆盖率,并且我也会不断检查该值。另一种方法是选择低于 100% 的值,但这样会出现上面提到的相同问题:其他代码也可能不会被测试覆盖,并且可能会被遗漏。
话虽如此,100% 的代码覆盖率当然不能保证您的代码没有任何错误。但是,如果您的应用程序代码中确实有未覆盖的行,您甚至不会对测试进行更改以发现该行中的潜在错误。
写出好的断言
编写测试的原因是我们想要断言代码的某种行为。因此断言是测试中非常重要的一部分。
当然,编写断言时最重要的考虑因素是它正确地测试代码的行为。但紧随其后的是代码失败时断言的行为方式。如果断言由于某种原因失败,那么问题对于开发人员来说应该尽可能明显。在此 symfony 拉取请求中当前正在处理的情况就是这种情况显而易见的情况。 symfony 附带了一个assertresponsestatuscodesame 方法,它允许在功能测试中检查响应的状态代码:
<?php
declare(strict_types=1);
class logincontrollertest extends webtestcase
{
public function testformattributes(): void
{
$client = static::createclient();
$client->request('get', '/login');
$this->assertresponsestatuscodesame(200);
$this->assertselectorcount(1, 'input[name="email"][required]');
}
}
这个测试的问题是它在状态码不是 200 的情况下生成的输出。由于测试通常在开发环境中运行,symfony 在访问这个 url 时会返回一个错误页面,assertresponsestatuscodesame 方法会输出断言失败时的完整响应。这个输出非常长,因为它不仅返回 html,还返回 css 和 javascript,而且我的回滚缓冲区实际上太小,无法让我阅读整个消息。
这绝对是我迄今为止遇到的最糟糕的例子,但如果代码中使用了错误的断言,它也会很烦人。让我们看一下上面的assertselectorcount断言的输出,如果给定的选择器没有恰好产生一个元素,则该断言会失败并显示以下消息:
failed asserting that the crawler selector "input[name="email"][required]" was expected to be found 1 time(s) but was found 0 time(s).
它很好地了解了发生的问题。但是,断言也可以用不同的方式编写(不要在家里这样做!):
$this->asserttrue($client->getcrawler()->filter('input[name="email"][required]')->count() === 1);
有人可能会说这完全一样,因此使用哪种变体并不重要。这与事实相差甚远,因为如果电子邮件没有单个必填输入字段,则会出现以下消息:
failed asserting that false is true.
这根本没有帮助,无论谁致力于解决问题,首先都必须弄清楚问题到底是什么。这表明,始终应该使用合适的断言,并且 phpunit 附带了许多适合所有类型用例的断言。有时创建自定义断言甚至是有意义的。
近年来我看到越来越流行的一个相对较新的断言是快照测试。尤其是当开始从事前端项目时,它似乎有很大帮助。我过去经常将它与 react 一起使用。主要要点是您的测试看起来像这样:
import renderer from 'react-test-renderer';
import Component from '../Component';
it('renders correctly', () => {
const tree = renderer
.create(<Component />)
.toJSON()
;
expect(tree).toMatchSnapshot();
});
神奇的事情发生在 tomatchsnapshot 方法中。在第一次运行时,它将树变量的内容写入单独的文件中。在后续运行中,它将树值的新值与之前存储在单独文件中的值进行比较。如果某些内容发生更改,它将导致测试失败并显示差异,并可以选择再次更新快照,这意味着您可以立即修复测试。
虽然这听起来确实不错,但它也有一些缺点。首先,快照非常脆弱,因为每当组件的渲染标记发生更改时,测试就会失败。其次,测试的意图是隐藏的,因为它没有解释作者真正想要测试的内容。
但是,我真正喜欢它的是,每当我更改组件时,它都会提醒我使用该组件的所有其他组件,因为所有这些快照在下次运行时都会失败。出于这个原因,我喜欢每个组件至少进行一次快照测试。
结论
总而言之,我认为您可以立即开始做一些事情来提高测试质量:
- 将测试中的代码保持在绝对需要的最低限度
- 目标是 100% 的代码覆盖率,如果无法测试,请正确地将代码从代码覆盖率机制中排除
- 当测试失败时,使用正确的断言可以获得更好的错误消息
在我看来,遵循这几条规则已经会产生巨大的影响,并帮助您长期享受在代码库中工作的乐趣!
以上就是《编写高质量的测试》的详细内容,更多关于的资料请关注golang学习网公众号!
在电脑上畅玩《王者荣耀》:最佳模拟器推荐与设置攻略
- 上一篇
- 在电脑上畅玩《王者荣耀》:最佳模拟器推荐与设置攻略
- 下一篇
- uniapp每日签到功能如何实现:后端PHP与前端uniapp的交互详解?
-
- 文章 · php教程 | 3分钟前 |
- PHP验证码生成教程实战详解
- 357浏览 收藏
-
- 文章 · php教程 | 24分钟前 | php 断点调试
- PHP代码调试技巧与断点设置详解
- 262浏览 收藏
-
- 文章 · php教程 | 29分钟前 | php如何加密解密
- Zend加密文件解密方法详解
- 297浏览 收藏
-
- 文章 · php教程 | 1小时前 |
- LIMIT分页怎么用?详解分页算法
- 496浏览 收藏
-
- 文章 · php教程 | 1小时前 |
- Symfony多对多关联与用户动态管理技巧
- 290浏览 收藏
-
- 文章 · php教程 | 1小时前 | php怎么运行
- PHP文件命令行执行方法详解
- 479浏览 收藏
-
- 文章 · php教程 | 1小时前 | PHP递归函数
- PHP递归解析嵌套配置文件方法
- 277浏览 收藏
-
- 文章 · php教程 | 2小时前 |
- Laravel父子表查询与多条件筛选教程
- 479浏览 收藏
-
- 文章 · php教程 | 2小时前 |
- PHP数据库备份优化方案详解
- 494浏览 收藏
-
- 文章 · php教程 | 2小时前 |
- PHP展示数据库视频路径的实用方法
- 354浏览 收藏
-
- 文章 · php教程 | 2小时前 | php sql注入
- PHP防范SQL注入方法及原理解析
- 414浏览 收藏
-
- 文章 · php教程 | 2小时前 |
- PHP防视频盗链的实用技巧与方法
- 379浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3319次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3531次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3563次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4682次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3936次使用
-
- PHP技术的高薪回报与发展前景
- 2023-10-08 501浏览
-
- 基于 PHP 的商场优惠券系统开发中的常见问题解决方案
- 2023-10-05 501浏览
-
- 如何使用PHP开发简单的在线支付功能
- 2023-09-27 501浏览
-
- PHP消息队列开发指南:实现分布式缓存刷新器
- 2023-09-30 501浏览
-
- 如何在PHP微服务中实现分布式任务分配和调度
- 2023-10-04 501浏览

