NUnit:如何从非静态方法传递 TestCaseData?

NUnit: How to pass TestCaseData from a non-static method?

我的测试失败是因为: 消息:在 TestCaseSourceAttribute 上指定的 sourceName 必须引用静态字段 属性 或方法。

这是我的代码:

const double MAX_DELTA = 0.01;
Qv_ges qv_ges_NE;
double Sum_Qv_ges_R_FL;
Qv_ges Qv_ges_Quer;

[SetUp]
public void init()
{
    qv_ges_NE = Din1946.Calc_Qv_ges_NE(205.7d);
    Sum_Qv_ges_R_FL = 15d + 15d + 15d + 15d + 15d + 10d + 10d + 10d + 10d + 10d + 10d + 10d;
    Qv_ges_Quer = Din1946.Calc_Qv_ges_Quer(qv_ges_NE, Sum_Qv_ges_R_FL);
}

public IEnumerable<TestCaseData> TestCases_A()
{
        yield return new TestCaseData(72.5, Qv_ges_Quer.FL.Value, MAX_DELTA);
        yield return new TestCaseData(169.17, Qv_ges_Quer.RL.Value, MAX_DELTA);
        yield return new TestCaseData(241.67, Qv_ges_Quer.NL.Value, MAX_DELTA);
        yield return new TestCaseData(314.17, Qv_ges_Quer.IL.Value, MAX_DELTA);
    }


    [TestCaseSource("TestCases_A")]
public void MethodA(double expected, double value, double latitude)
      { Assert.AreEqual(expected, value, latitude);}

我只使用了静态测试用例方法,但现在我需要这样的数据: Qv_ges_Quer.IL.Value, Qv_ges_Quer.FL.Value.. 所以我删除了 static

如何使用非静态测试用例?通过调试我也注意到它一开始并没有进入SetUp

这是我想重新组织的旧代码,我想你知道 another/better 方式然后是上面的方式:

public void MethodA(){
  Qv_ges qv_ges_NE = Din1946.Calc_Qv_ges_NE(205.7d);

  double Sum_Qv_ges_R_FL = 15d + 15d + 15d + 15d + 15d + 10d + 10d + 10d + 10d + 10d + 10d + 10d;
  Qv_ges Qv_ges_Quer = Din1946.Calc_Qv_ges_Quer(qv_ges_NE, Sum_Qv_ges_R_FL);

  Assert.AreEqual(72.5, Qv_ges_Quer.FL.Value, MAX_DELTA);
  Assert.AreEqual(169.17, Qv_ges_Quer.RL.Value, MAX_DELTA);
  Assert.AreEqual(241.67, Qv_ges_Quer.NL.Value, MAX_DELTA);
  Assert.AreEqual(314.17, Qv_ges_Quer.IL.Value, MAX_DELTA);
}

根据设计,方法 属性 或 TestCaseSourceAttribute 使用的字段必须是静态的。这是为了避免在加载测试时需要实例化夹具 class。您的夹具仅在我们启动 运行 时实例化 - 在 GUI 的情况下,每次我们启动 运行 - 它的生命周期仅与 运行夹具.

就您而言,您似乎发现可以使用静态方法。如果可能的话,那是最好的。

此处使用实例方法的唯一方法是使用构造函数 TestCaseSourceAttribute(Type sourceType),其中 sourceType 直接实现 IEnumerable 和 returns 您的测试用例数据。如果您使用它,我建议使用与您的 TestFixture 不同的 class。这不是绝对必要的。如果您使用相同的 class,将在加载时和 运行 时创建不同的实例,它们之间没有任何联系。许多开发人员最终对此感到困惑,并试图在加载时留下状态以供测试使用。那不行。

‍♀️ 僵尸反应,迟到总比不到好

实现此目的的另一种方法是让您的测试用例数据源 return 接受您需要的非静态成员作为其参数的函数对象。然后您的测试调用它来创建您希望 NUnit 可以传递给您的数据。

在你的情况下,它看起来像:

private static IEnumerable<TestCaseData> GetTestDataA()
{
    yield return new TestCaseData(72.5,   new Func<Qv_ges, double>( qvGesQuer => qvGesQuer.FL.Value ), MAX_DELTA);
    yield return new TestCaseData(169.17, new Func<Qv_ges, double>( qvGesQuer => qvGesQuer.RL.Value ), MAX_DELTA);
    yield return new TestCaseData(241.67, new Func<Qv_ges, double>( qvGesQuer => qvGesQuer.NL.Value ), MAX_DELTA);
    yield return new TestCaseData(314.17, new Func<Qv_ges, double>( qvGesQuer => qvGesQuer.IL.Value ), MAX_DELTA);
}

[TestCaseSource( nameof(GetTestDataA) )]
public void MethodA( double expected, Func<Qv_ges, double> getValue, double latitude)
{ 
    Assert.AreEqual( expected, getValue( Qv_ges_Quer ), latitude );
}

如果您需要多个参数,请将它们添加到仿函数和 lambda 的参数中,或者考虑传入 this。如果您需要多个 return 值,请将函数对象 return 设为元组:

new Func<Qv_ges, (double, double)>( qvGesQuer => (qvGesQuer.RL.Value, qvGesQuer.IL.Value) )

另一种方法是传入 nameof() 字符串作为测试参数,并使用反射来获取这些参数的值。