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()
字符串作为测试参数,并使用反射来获取这些参数的值。
我的测试失败是因为: 消息:在 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()
字符串作为测试参数,并使用反射来获取这些参数的值。