1

相对的问题可能像this 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
    }
  }
}

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

4

2 回答 2

4

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

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

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

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

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

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

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

于 2013-04-25T08:22:30.127 回答
4

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

因此,您应该做的是对所有类进行单元测试,并且由于您正在练习依赖注入,因此伪造被测试的类依赖项应该很容易。

于 2013-04-25T09:11:54.390 回答