Moq,使用构造函数依赖调用 base

Ant*_*alo 3 .net c# unit-testing moq mocking

我有一堂课

 public class Foo 
 {
      public Foo(IBar bar)
      {
         //...
      }
      public virtual int GetValue(){}
      public virtual DoActual()
      {
           ...
         var value = GetValue();
           ...
      }
 }
Run Code Online (Sandbox Code Playgroud)

我需要测试一下。我想覆盖 GetValue 方法并将特定值返回给调用者。这是我的测试代码:

[TestMethod]
public void Test()
{
    var mock = new Mock<Foo>{ CallBase=true}; // problem is here
    mock.Setup(x=>x.GetValue()).Returns(1); 

    mock.DoActual();

 //asserts
}
Run Code Online (Sandbox Code Playgroud)

由于我的 Foo 类构造函数具有依赖项,因此我必须将其传递给对象初始值设定项。Moq 提供将参数传递给 params[] 重载:

[TestMethod]
public void Test()
{
   var barMock= new Mock<IBar>();
    var mock = new Mock<Foo>(barMock){ CallBase=true}; // problem is still here
    mock.Setup(x=>x.GetValue()).Returns(1); 
    mock.DoActual();
 //asserts
}
Run Code Online (Sandbox Code Playgroud)

当我以这种方式通过我的模拟时,我收到错误消息,说 Moq.Mock`1[IBar] 不是 Foo 接受的构造函数。

当我将 barMock.Object 传递给构造函数时,我得到了不同类型的基本相同的错误:

"Could not find a constructor that would match given arguments:
 Castle.Proxies.IBarProxy"
Run Code Online (Sandbox Code Playgroud)

我被困在这个地方,不知道如何解决我的 Foo 类依赖项。

for*_*rir 5

您上面最小化的问题无法编译,这表明您尚未以当前形式对其进行测试。这使得很难判断您的实际问题是否得到了正确的描述,或者您是否将问题缩小了太多并删除了您的实际问题,在这种情况下,您可能需要编辑您的问题以包含其他详细信息。下面的代码,基于上面的编译和工作。

public interface IBar {
}

public class Foo 
{
   public Foo(IBar bar)
   {
       //...
   }
   public virtual int GetValue(){ return 0;}
   public virtual void DoActual()
   {
       var value = GetValue();
   }
}

[TestFixture]
public class SomeTests {
    [Test]
    public void TestMethodCalled() {
        var barMock = new Mock<IBar>();
        var mock = new Mock<Foo>(barMock.Object) { CallBase = true };
        mock.Setup(x => x.GetValue()).Returns(1);
        mock.Object.DoActual();
        //asserts
        mock.Verify(x => x.GetValue());
    }
}
Run Code Online (Sandbox Code Playgroud)

它传递barMock.Object给构造函数,并验证该方法是否已被调用。

如前所述,模拟您正在测试的类通常是一个坏主意,更好的方法是操作/模拟其依赖项,GetValue以便它返回您所追求的结果。显然,您执行此操作的能力将取决于该方法的作用以及必要时您可以更改多少代码,有时,尤其是对于遗留代码,您只需要使用已有的代码即可。