5

我正在尝试模拟对象的一个​​属性

有一个类似的问题: 返回返回另一个替代的方法的结果会在 NSubstitute 中引发异常 但是接受的答案对我不起作用。

void Main()
{
    var obj = Substitute.ForPartsOf<MyObject>();
    //WORKS, But I need a partial mock!:
    //var obj = Substitute.For<MyObject>();

    obj.PropClass.Returns(Substitute.For<PropClass>());

    //It's suggestion, Fails, same error:
    //var returnValue = Substitute.For<PropClass>();
    //obj.PropClass.Returns(returnValue);

    //Fails, same error:
    //Lazy implementation of *similar question*
    //Func<PropClass> hello = () => Substitute.For<PropClass>();
    //obj.PropClass.Returns(x => hello());

    //Fails, same error:
    //I believe what *similar question* suggests
    //obj.PropClass.Returns(x => BuildSub());

    obj.PropClass.Dump("Value");
}

public class MyObject
{
    public MyObject()
    {
        _propClasses = new List<PropClass>();
    }
    private readonly IList<PropClass> _propClasses;
    public virtual IEnumerable<PropClass> PropClasses { get { return _propClasses; } }
    public virtual PropClass PropClass { get { return PropClasses.FirstOrDefault(); } }
}
public class PropClass
{
}

public PropClass BuildSub()
{
    return Substitute.For<PropClass>();
}

这些失败,但有以下例外:

CouldNotSetReturnDueToTypeMismatchException:
Can not return value of type PropClassProxy_9 for MyObject.get_PropClasses (expected type IEnumerable`1).

Make sure you called Returns() after calling your substitute (for example: mySub.SomeMethod().Returns(value)),
and that you are not configuring other substitutes within Returns() (for example, avoid this: mySub.SomeMethod().Returns(ConfigOtherSub())).

If you substituted for a class rather than an interface, check that the call to your substitute was on a virtual/abstract member.
Return values cannot be configured for non-virtual/non-abstract members.

Correct use:
  mySub.SomeMethod().Returns(returnValue);

Potentially problematic use:
  mySub.SomeMethod().Returns(ConfigOtherSub());
Instead try:
  var returnValue = ConfigOtherSub();
  mySub.SomeMethod().Returns(returnValue);
4

1 回答 1

9

好的,这有点棘手。首先,解决方案是停止obj.PropClass调用基础实现:

obj.When(x => { var get = x.PropClass; }).DoNotCallBase();
obj.PropClass.Returns(prop);

现在解释。NSubstitute 记录对替代品的调用,当我们调用 时Returns,它会抓取最后一次调用并尝试将其配置为返回特定值。

运行时发生的情况是调用obj.PropClass.Returns(prop)了 real obj.PropClass,而后者又调用了obj.PropClasses,因此 NSubstitute 现在认为obj.PropClasses是最后一次调用。The Returnsthen 尝试在期望aPropClass时返回一个,因此出现异常。PropClassesIEnumerable<PropClass>

上面的修复程序停止obj.PropClass立即调用基本实现,因此最后一个调用不会被替换,PropClasses并且Returns可以按预期工作。

诚然,这非常糟糕,也是我们长期坚持将部分模拟放入 NSubstitute 的原因之一。NSub 良好的语法的代价是它并不总是能够区分开发人员何时配置调用和何时需要运行真实代码。对困惑感到抱歉。

于 2014-01-23T00:10:12.457 回答