10

我正在尝试在 C# 中模拟 System.net.Sockets.Socket 类 - 我尝试使用 NUnit 模拟,但它不能模拟具体类。我也尝试使用 Rhino Mocks,但它似乎使用了该类的真实版本,因为它在调用 Send(byte[]) 时抛出了 SocketException。有没有人使用任何模拟框架成功创建和使用了 Socket 模拟?

4

5 回答 5

15

每当我在 Moq 中遇到这类问题时,我最终都会创建一个接口来抽象出我无法模拟的东西。

因此,在您的实例中,您可能有一个实现 Send 方法的 ISocket 接口。然后让你的模拟框架模拟它。

在您的实际代码中,您将拥有这样的类

public class MySocket : ISocket
{
  System.Net.Sockets.Socket _socket;

  public void MySocket(System.Net.Sockets.Socket theSocket)
  {
    _socket = theSocket;
  }

  public virtual void Send(byte[] stuffToSend)
  {
    _socket.Send(stuffToSend);
  }

}

不确定这是否满足您的需求,但这是一种选择。

于 2009-08-12T00:07:05.923 回答
3

上面的类只模拟你的发送方法。这实际上模拟了一个 Socket。它继承了所有的 Socket,然后实现了 ISocket 接口。ISocket 需要实现你需要模拟的任何 Socket 方法或属性的签名

//internal because only used for test code
internal class SocketWrapper : Socket, ISocket
{
    /// <summary>
    /// Web Socket
    /// </summary>
    public SocketWrapper():base(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp)
    {
    }

    //use all features of base Socket
}

界面看起来像这样,有两个方法被拒绝:

public interface ISocket
{
    void Connect(IPAddress address, int port);
    int Send(byte[] buffer, int offset, int size, SocketFlags socketFlags, out SocketError errorCode);
}

使用它们的类有 2 个构造函数。一个注入一个 ISocket 进行测试,然后一个注入应用程序使用的自己的 Socket。

public class HTTPRequestFactory3
{
internal ISocket _socket;


    /// <summary>
    /// Creates a socket and sends/receives information.  Used for mocking to inject ISocket
    /// </summary>
    internal HTTPRequestFactory3(ISocket TheSocket)
    {
     _socket = TheSocket as ISocket;
     this.Setup();
    }

    /// <summary>
    /// Self Injects a new Socket.
    /// </summary>
    public  HTTPRequestFactory3()
    {
        SocketWrapper theSocket = new SocketWrapper();
        _socket = theSocket as ISocket;
        this.Setup();
    }
}

然后您的测试可以创建一个 ISocket,设置期望并验证它们是否运行上述类将与真实套接字一起使用的所有相同代码。此测试验证该部分代码。

   [Test]
   public void NewSocketFactoryCreatesSocketDefaultConstructor()
        {
            webRequestFactory = new HTTPRequestFactory3();
            Assert.NotNull(webRequestFactory._socket);
            Socket testSocket = webRequestFactory._socket as Socket;
            Assert.IsInstanceOf<Socket>(testSocket);
        }
于 2013-06-24T12:03:18.510 回答
2

调用 Send 方法时收到 SocketException 的原因是因为 Send 不是可覆盖的方法。为了使 RhinoMocks 能够模拟属性或方法的行为,它必须在接口中定义(然后我们创建我们的模拟)或者是可覆盖的。

您对此的唯一解决方案是创建一个可模拟的包装类(如 thinkzig 所建议的)。

于 2009-08-12T09:36:10.423 回答
1

我使用 thinkzig 建议的方法(Socket 的适配器类)创建了一个示例控制台应用程序。它使用 RhinoMocks 和 NUnit。你可以在这里下载它:如何模拟 System.Net.Sockets.Socket

于 2009-12-28T17:15:04.680 回答
0

您最好创建一个接口并在您的测试中模拟它,并在您的代码中实现一个包装类,如 thinkzig 所说,将所有方法调用转发到 .NET 套接字。看看这个链接,它是同样的问题:你如何在 C# 中模拟文件系统以进行单元测试?

于 2009-08-12T04:57:44.523 回答