当前位置: 首页 > news >正文

携程总监的单元测试是怎么样写的?

 

大家都知道,开发软件的时候为代码编写单元测试是很好的。但实际上,光有测试还不够,还要编写好的测试,这同样重要。

要做到这一点,考虑遵循一些固执的原则,对测试代码给予一些关爱:

1. 保持测试代码的紧凑和可读性

要做到这一点,应该要进行毫不留情的重构,就像对生产代码应该做的那样。否则让测试代码随着时间腐化,就是在测试里面制造可怕的遗留代码。如果测试不能很容易重构,那么生产代码也很难重构,从而导致生产系统的遗留代码。始终做一个勇敢的重构者。

2. 避免编写重复累赘的断言

举个例子,测试代码使用正则表达式生成内容,而这个正则表达式是跟生产代码的解析器中使用的一模一样的。

一般来说,我们不希望在测试和代码之间复制逻辑。因此,在测试中复制正则表达式或其他内容不是一种选择。在这种情况下,考虑测试输入激励/输出结果之间的关系(f(输入) - >输出)可能会有帮助,例如,如果代码的目标是要做模板替换,不要在测试代码里用原始值来做替换。相反,在测试里面直接指定预期的计算结果。

// 使用
Assertions.assertThat(processTemplate("param1", "param2")).isEqualTo("this is 'param1', and this is 'param2'"));// 而不要用
Assertions.assertThat(processTemplate("param1", "param2")).isEqualTo("this is '%s', and this is '%s'", param1, param2));
复制代码

3. 覆盖尽可能多的范围,包括正面情况,以及(甚至更重要的)出错的代码路径。

通常,要做到这一点,最好的办法试采用测试驱动开发(Test Driven Development)。通过TDD,人们可以在设计时识别可能会出错的部分。不要羞于为一段小代码编写一个简单的测试用例。你永远不知道什么时候,为什么以及以什么方式,你会要用到甚至修改这段代码。

可以研究一下如何检查测试的有效性,类似PIT这样的工具可以进行变更测试,值得研究一下。

4. 不要Mock你不拥有的类型!

这不是一个硬界限,但越过这条线很可能会产生反作用力!

TDD是关于设计的,也是关于测试的,两者一样重要,在模拟外部API时,测试不能用于驱动设计,API属于第三方;这个第三方可以,并且实际上也经常会更改API的签名和行为。

想象一下Mock第三方Lib的代码。在第三方库的某次升级之后,它的逻辑可能会改变,但测试套件仍会执行得很好,因为它被Mock了。所以后来,你认为一切都很好,毕竟构建墙是绿色的,软件部署上去,然后......嘣

如果你感觉需要Mock第三方库,可能表明你当前的设计与第三方库没有足够的分离。

另一个问题是第三方库可能很复杂,需要大量的Mock才能正常工作。这导致过度指定的测试和复杂的测试辅助装置,这本身就损害了紧凑和可读的目标。或者由于模拟外部系统过于复杂,从而导致测试代码对生产代码的覆盖不足。

取而代之的最常见的方法,是围绕外部lib / 系统创建包装器,尽管应该意识到抽象泄漏的风险,其中过多的低级API,概念或异常超出了包装器的边界。为了验证与第三方库的集成,编写集成测试,并使它们尽可能紧凑和可读。

5. 不要Mock一切,这是一种反模式

如果一切都被Mock,我们真的在测试生产代码吗?该不Mock的时候,不要犹豫!

不要Mock值对象

为什么人们甚至想要这样做?

因为实例化对象太痛苦了! => 不是正当理由。

如果创建新的对象太难了,那么代码可能需要一些严肃的重构。另一种方法是为您的值对象创建构建器 - 有一些工具,包括IDE插件,Lombok和其他。还可以在测试类路径中创建有意义的工厂方法。

abstract class CustomerCreations {public static Customer customer_with_a_single_item_in_the_basket() {// long init sequence}
}
复制代码

 Mockito专注于对象之间的相互操作,这是面向对象编程的核心部分。

http://www.lryc.cn/news/58369.html

相关文章:

  • 算法每日一题:P2089 烤鸡 -DFS练习
  • Spring中的循环依赖是什么?如何解决它?
  • 不良事件报告系统源码,PHP医院安全(不良)事件报告系统源码,在大型医院稳定运行多年
  • MySQL 查询常用操作(3)——排序 order by
  • Android Jetpack 从使用到源码深耕【数据库注解Room 从实践到原理 】(二)
  • 传统企业如何实现数字化转型?
  • Linux修改密码报错Authentication token manipulation error的终极解决方法
  • ROS实践06 自定义消息类型
  • 《剑指offer》——从尾到头打印链表
  • Javaweb基础配置模板(mybatis+javaweb)
  • 物联网 JS 前端框架开发 - 执行 js 程序
  • 区块链概论
  • MAC地址表安全
  • 处理CSV(python)
  • 【云原生】Kubernetes(k8s)之容器的探测
  • 看完这个你就牛了,自动化测试框架设计
  • Spring Cloud Alibaba全家桶(八)——Sentinel规则持久化
  • Mysql不锁表备份之Xtrabackup的备份与恢复
  • flex布局:输入框布局demo
  • PHP请求的好处,PHP如何请求淘宝开放接口
  • 精选出来的几道Java语法基础面试题
  • uniapp或者小程序图片选择中的sizeType属性到底是什么
  • 判断一个字符串是否是回文
  • 国产软件爆发!中国版Navicat,SQL Studio成数据库管理工具热门
  • 算法学习day51
  • 10 JS01——初识JS
  • 【软考备考-综合知识】安全性、可靠性与系统性能评测基础知识
  • 匆忙之间难免疏忽,写代码更加如此
  • 低代码(七)低代码平台后端技术选型2.0
  • UDS介绍