【问题标题】:What to be Tested in Layered Dependency Injection分层依赖注入要测试什么
【发布时间】:2013-04-25 06:05:37
【问题描述】:

相对问题可能是这样的one question

我使用依赖注入作为我的应用程序的架构。然后我为它创建了单元测试。

注入架构可能是这样的:

IClassA(IClassB)
IClassB(IClassC1, IClassC2)
//more of it

请注意,此架构仅涉及服务对象,不涉及存储库。现在我想知道,要测试什么类。

  • 如果我测试 C1 和 C2 类(最小的类),则不测试 B 类和 A 类。

  • 如果我只测试A类(集成类),那么需要覆盖的场景太多,而不是很多小模块。

  • 如果我测试 A 类的模块,同时测试 C1 和 C2 类,我认为这似乎是多余的。如果我想重构一个逻辑(在开发期间),我需要管理许多测试单元。

  • 如果我用模拟 B 类测试 A 类,它将为每个类和模拟类创建几乎 1:1 的比例。不会造成太多的嘲讽吗?

任何建议或想法将不胜感激。

编辑:

当我想提供文件信息(基于 csv 或 xml),然后转换为实体时,这是一个工作场景。该过程将是:

  • ClassA 读取数据,将其作为大表格式(可能在 DataTable 中)传递给 ClassB
  • ClassB 使用 ClassC1 进行验证
  • ClassB 使用 ClassC2 进行更多验证,然后返回标头-详细信息映射实体。

类的示例代码将是这样的(我跳过了构造函数注入部分):

public class ClassA: IClassA{
  public IEnumerable<Request> GetRequestFromFile(FileInfo info
    , ref ValidationResult validationResult){
    //read the file and get DataTable
    iClassB.ConvertToRequest(dataTableResult, ref validationResult);
  }
}

public class ClassB : IClassB{
  public IEnumerable<Request> ConvertToRequest(DataTable dt
    , ref ValidationResult validationResult){
    foreach(DataRow row in dt.Rows){
      // convert to flat request first, to avoid reading DataTable too much
      iClassC1.Validate(rawRequest, ref validationResult);
      iClassC2.Validate(rawRequest, ref validationResult);
    }
    if(validationResult.IsSuccess){
      // convert and return the header-detail entity object
    }
  }
}

注意:请忽略逻辑架构(例如:关于在验证期间不抛出异常等)

【问题讨论】:

  • 请提供一些示例代码。如果您测试 ClassA,我认为模拟 ClassB 是有意义的,这样您就不必依赖外部世界,这正是我们在单元测试中想要的。当你所有的单元测试最终都通过时,你应该考虑集成测试。
  • @MarcoForberg 我已经为此更新了一个真实的案例实现。也许它可以帮助理解这个案例。

标签: unit-testing testing dependency-injection automated-tests


【解决方案1】:

您应该始终模拟所有外部影响,因此您只测试一件事的功能。它可能需要大量的模拟,但这对于所有可用的模拟框架(如 Rhinomocks、Moq 和许多其他框架)来说并不是真正的问题。

遵循该规则,您应该始终:

  • 在测试 ClassA 时模拟 IClassB
  • 在测试 ClassB 时模拟 IClass1IClass2

如果您不模拟注入的对象,那么在重构期间您会遇到令人困惑的测试失败。此外,您的测试将是集成测试而不是单元测试,因为它们取决于外部类的实现。

示例:您正在ClassATesFixture 中测试ClassA。您决定使用生产代码ClassB 作为输入,而不是模拟IClassB。测试没问题。现在,您重构 ClassB,并引入一个错误。这将导致 ClassATesFixture 中的测试失败,尽管 ClassA 中没有任何变化。

因此,如果您在测试中使用真实对象作为输入,那么您实际上测试的不仅仅是您要测试的对象。最重要的是,您还会遇到令人困惑的测试失败。

理想情况下,除非测试类中的某些内容发生了变化,否则任何测试都不应该从绿色变为红色。 Mocking 是实现这一目标的方法。

【讨论】:

  • 感谢您的解释。真的很感激,我想我知道我必须做什么。我已经为这个真实案例提供了一个例子,也许它可以加强你的观点。
  • 根据我的经验,当您“需要大量模拟”时,您的应用程序设计就会出现问题。我构建的应用程序是围绕几个精心挑选的通用抽象设计的,这些抽象涵盖了系统中大约 70% 到 95% 的服务。这允许我只编写几个通用模拟对象(每个通用抽象一个),这使得在大多数情况下使用模拟框架是多余的。
  • @Steven:是的,如果你的类需要大量不同的输入,因此需要大量模拟,那么可能存在设计缺陷,但这不是模拟引入的缺陷。我只是说在我的单元测试中,相当大一部分代码是关于设置模拟的。
  • @Steven 目前我不使用模拟框架,并且由于几个原因目前我不能。不过感谢您的建议,我将尝试学习这些模拟框架以供进一步使用。
  • 两个答案都强调模拟注入的依赖项。我选择这个答案是因为它有示例并提到了一些自动模拟框架。
【解决方案2】:

我认为您误解了“单元测试”一词。单元测试意味着单独测试类。在您的情况下,您应该通过将虚假依赖项传递给要测试的类来做到这一点。如果您不在您测试的类中使用虚假依赖项,那么您就不会孤立地测试这些类,因此您不会进行单元测试。在这种情况下,您正在做的是集成测试。

所以你应该做的是对所有类进行单元测试,并且由于你正在练习依赖注入,所以在测试中伪造一个类依赖项应该很容易。

【讨论】:

  • 是的,看来我误解了单元测试和集成测试。感谢您指出这一点,现在我明白了我必须做什么和不应该做什么。
猜你喜欢
  • 2023-03-17
  • 1970-01-01
  • 2018-01-27
  • 1970-01-01
  • 2021-06-19
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多