在本系列的第一部分中,我们看到了有效测试应满足的一些普遍适用的原则和约束。 在这一部分中,我们将仔细研究代码级单元测试以及组件用例测试。

单元测试

单元测试验证单个单元(通常是类)的行为,而忽略或模拟该单元外部的所有问题。 单元测试应测试各个单元的业务逻辑,而不验证其进一步的集成或配置。

根据我的经验,大多数企业开发人员对单元测试的构建方式都有很好的了解。 您可以在我的咖啡测试项目中查看此示例,以了解想法。 大多数项目将JUnit与Mockito结合使用以模拟依赖关系,理想情况下使用AssertJ来有效定义可读的断言。 我一直认为,我们可以执行单元测试而无需特殊的扩展程序或运行程序,即仅使用纯JUnit运行它们。 原因很简单:执行时间; 我们应该能够在几毫秒内运行数百个测试。

单元测试通常执行速度非常快,并且易于执行,并且不会对测试套件的生命周期施加任何约束,因此可以轻松支持构建复杂的测试套件或特殊的开发工作流程。

但是,具有许多模拟被测类的依赖关系的单元测试的缺点是,它们将与实现紧密结合,尤其是类结构和方法,这使得重构代码变得困难。 换句话说,对于生产代码中的每个重构动作,测试代码也需要更改。 在最坏的情况下,这会导致开发人员进行较少的重构,这仅仅是因为它们变得太麻烦了,从而很快导致项目代码质量下降。 理想情况下,开发人员应该能够重构代码并四处移动,只要他们不改变应用程序的行为(从用户的角度来看)即可。 单元测试并不总是使重构生产代码变得容易。

根据项目经验,单元测试对于测试具有简洁逻辑或功能的高密度代码(例如特定算法的实现)非常有效,同时又不会与其他组件发生过多交互。 特定类中的代码密度越小或越复杂,循环复杂性越低,或者与其他组件的交互作用越高,则测试该类的单元测试效果就越差。 尤其是在具有少量专业业务逻辑并且与外部系统的集成程度相当的微服务中,可以说,对许多单元测试的需求减少了。 除了少数例外,这些系统的各个单元通常包含很少的专用逻辑。 选择在哪里花时间和精力进行权衡时,必须考虑到这一点。

用例测试

为了解决将测试与实现紧密耦合的问题,我们可以使用略有不同的方法来扩大测试范围。 在我的书中 ,我描述了组件测试的概念,因为缺少更好的术语,我们也可以将其称为用例测试。

用例测试是代码级别的集成测试,由于测试启动时间的原因,它们还没有使用嵌入式容器或反射扫描。 他们验证通常参与单个用例的一致组件的业务逻辑行为,从边界的业务方法一直到所有涉及的组件。 与外部系统(如数据库)的集成已被嘲笑。

在不使用更先进的技术自动建立组件连接的情况下构建此类方案听起来很费力。 但是,我们定义了可重用的测试组件或test double ,它们通过模拟,接线和测试配置来扩展组件,以最大程度地减少重构变更的整体工作量。 目标是制定单一职责,以将变更的影响限制在测试范围内的单个或几个类中。 以可重复使用的方式执行此操作会限制总体所需的工作量,并且在项目规模变大时会得到回报,因为我们每个组件只需支付一次管道费用,这很快就可以摊销。

为了获得更好的主意,假设我们正在测试订购咖啡的用例,其中包括两个类CoffeeShopOrderProcessor


测试双重类CoffeeShopTestDoubleOrderProcessorTestDouble*TD驻留在项目的测试范围中,而它们扩展了驻留在主要范围中的CoffeeShopOrderProcessor组件。 测试双打可能会设置所需的模拟和连线逻辑,并可能使用与用例相关的模拟或验证方法来扩展类的公共接口。

下面显示了CoffeeShop组件的测试double类:

 public class CoffeeShopTestDouble extends CoffeeShop { public CoffeeShopTestDouble(OrderProcessorTestDouble orderProcessorTestDouble) { entityManager = mock(EntityManager. class ); orderProcessor = orderProcessorTestDouble; } public void verifyCreateOrder(Order order) { verify(entityManager).merge(order); } public void verifyProcessUnfinishedOrders() { verify(entityManager).createNamedQuery(Order.FIND_UNFINISHED, Order. class ); } public void answerForUnfinishedOrders(List<Order> orders) { // setup entity manager mock behavior }  } 

测试double类可以访问CoffeeShop基类的字段和构造函数以设置依赖项。 它使用其测试双重形式的其他组件(例如OrderProcessorTestDouble )来能够调用用例中包含的其他模拟或验证方法。

测试双重类是可重用的组件,在每个项目范围内编写一次,并在多个用例测试中使用

 class CoffeeShopTest { private CoffeeShopTestDouble coffeeShop; private OrderProcessorTestDouble orderProcessor; @BeforeEach void setUp() { orderProcessor = new OrderProcessorTestDouble(); coffeeShop = new CoffeeShopTestDouble(orderProcessor); } @Test void testCreateOrder() { Order order = new Order(); coffeeShop.createOrder(order); coffeeShop.verifyCreateOrder(order); } @Test void testProcessUnfinishedOrders() { List<Order> orders = Arrays.asList(...); coffeeShop.answerForUnfinishedOrders(orders); coffeeShop.processUnfinishedOrders(); coffeeShop.verifyProcessUnfinishedOrders(); orderProcessor.verifyProcessOrders(orders); }  } 

用例测试验证在入口点(这里为CoffeeShop上调用的单个业务用例的处理。 这些测试变得简短且易读,这是因为接线和模拟发生在单个测试双打中,并且它们还可以利用特定于用例的验证方法,例如verifyProcessOrders()

如您所见,测试双重扩展了生产范围类,用于设置模拟和验证行为的方法。 尽管这似乎是一些设置工作,但如果我们有多个用例可以在整个项目中重用组件,则成本将快速摊销。 我们的项目增长得越多,这种方法的好处就越大,尤其是当我们查看测试执行时间时。 我们所有的测试用例仍然使用JUnit运行,它可以立即执行数百个测试用例。

这是这种方法的主要优点:用例测试的运行速度与普通单元测试一样快,但由于仅需对单个或几个组件进行更改,因此可以方便地重构生产代码。 此外,使用针对我们领域的富有表现力的设置和验证方法来增强测试效率,从而使我们的测试代码更具可读性,便于使用,并避免了测试案例中的样板代码。

不包含任何高级测试上下文运行程序的代码级测试可以非常快速地执行,并且即使在非常复杂的项目中也不会为整体构建增加太多时间。 该系列的下一部分将显示代码级以及系统级集成测试。

翻译自: https://www.javacodegeeks.com/2019/09/efficient-enterprise-testing-unit-use-case.html

高效的企业测试-单元和用例测试(2/6)相关推荐

  1. 如何设计高效测试用例_高效的企业测试-单元和用例测试(2/6)

    如何设计高效测试用例 在本系列的第一部分中,我们看到了有效测试应满足的一些普遍适用的原则和约束. 在这一部分中,我们将仔细研究代码级单元测试和组件或用例测试. 单元测试 单元测试验证单个单元(通常是类 ...

  2. 高效的磁力搜索引擎 -_高效的企业测试-结论(6/6)

    高效的磁力搜索引擎 - 该系列的最后一部分将涵盖其他端到端测试,生产中的测试以及各部分的结论. 进一步的端到端测试和生产中的测试 除了仅验证单个被测应用程序并模拟外部问题的系统测试之外,我们的管道还必 ...

  3. 高效的企业测试-结论(6/6)

    该系列的最后一部分将涵盖其他端到端测试,生产中的测试以及各部分的结论. 进一步的端到端测试和生产中的测试 除了仅验证单个被测应用程序并模拟外部问题的系统测试之外,我们的管道还必须包括完整的端对端测试, ...

  4. 高效的企业测试-集成测试(3/6)

    本系列的这一部分将展示如何通过代码级以及系统级集成测试来验证我们的应用程序. (代码级)集成测试 集成测试一词有时在不同的上下文中使用不同. 根据Wikipedia的定义,我指的是在代码级别上验证多个 ...

  5. 测试框架 如何测试私有方法_高效的企业测试–测试框架(5/6)

    测试框架 如何测试私有方法 本系列文章的这一部分将介绍测试框架以及我在何时以及是否应用它们方面的想法和经验. 关于测试框架的想法 我对大多数测试框架不太满意的原因是,按照我的观点,它们大多增加了语法上 ...

  6. 高效的企业测试–测试框架(5/6)

    本系列文章的这一部分将介绍测试框架以及我在何时以及是否应用它们方面的想法和经验. 关于测试框架的思考 我对大多数测试框架不太满意的原因是,按照我的观点,它们大多增加了一些语法上的便利性和便利性,但本质 ...

  7. 高效的敏捷测试第十一课 敏捷测试分析、策略和方法

    第26讲:基于上下文驱动思维的测试分析 从这一讲开始,我们就进入了第 5 部分内容的学习:敏捷测试分析与计划.在这一部分你将学到:测试需求分析.测试风险的识别.测试策略及测试计划的制定.今天先从基于上 ...

  8. mac 爱普生打印机驱动_高效打印企业首选 爱普生M2178黑白多功能一体机评测

    提到打印机相信每一个公司都配备了一台,因为它是日常办公中不可缺少.尽管现在很多的工作流程都已经可以在线上执行,且提倡无纸化办公,但很多重要的材料还是需要打印出来存档.签字.因此打印机还是一个非常必要的 ...

  9. 第七章 项目进阶,构建安全高效的企业服务

    第七章 项目进阶,构建安全高效的企业服务 1.Spring Security 1.1 基本介绍 Spring Security是一个能够为基于Spring的企业应用系统提供声明式的安全访问控制解决方案 ...

最新文章

  1. 攻击面管理预防网络攻击原理?
  2. 蚂蚁金服核心技术:百亿特征实时推荐算法揭秘
  3. python大量数据折线图-Python数据可视化练习:各种折线图的用法
  4. /dev/mapper/VolGroup00-LogVol00 100% 如何处理
  5. linux   图片
  6. linux的yum详解,Linux之YUM 详解
  7. mysql udf安全_打造全功能MYSQL入侵UDF
  8. TeXstuidio没有中文的拼写检查
  9. grafana安装配置 influxdb安装配置
  10. Java并发包-原子类
  11. ibm服务器aix系统查看cpu,IBM AIX系统硬件信息查看命令(shell脚本)
  12. MyBatis简介及下载地址
  13. 投标是个技术活,不这样做要么苟且,要么狗带
  14. ansys命令流——点线面体基础操作(02)
  15. HHL算法第四弹(回顾伴随、正定算子、半正定算子、正规算子、酉矩阵、幺正矩阵、厄米矩阵,极式分解,奇异值分解)
  16. RLS实现求解最小二乘确定性正则方程
  17. [论文阅读]PIXER: an automated particle-selection method based on segmentation using deep neural network
  18. 智能车学习日记【四】————环岛
  19. FreeNOS--项目的README
  20. 网络直播课程:神马是敏捷?(直播时间:2014-7-14 20:00-21:00)

热门文章

  1. codeforces1484 B. Restore Modulo(数学)
  2. 深度学习中反向传播算法简单推导笔记
  3. codeforces E. Game with String 概率
  4. Spark入门(二)多主standalone安装
  5. 面试官让我讲讲Java中的锁,我笑了
  6. Java开发2018年值得学习的10大技术
  7. Maven精选系列--依赖范围、传递、排除
  8. Java Jar包的压缩、解压使用指南
  9. laravel关闭crsf
  10. 数组的初始化与默认值