套接字异步发送和接收数据

时间:2013-07-19 18:59:06

标签: .net multithreading sockets asynchronous tcp

我正在开发一种从大量套接字发送和接收数据的软件。它是高度异步的,因此我担心以下情况:N个线程试图将数据发送到同一个套接字。这是,N个线程同时调用BeginSend。

澄清:我没有使用线程,但'系统'会这样做。这就是文档所说的:“当你的应用程序调用BeginSend时,系统将使用一个单独的线程来执行指定的回调方法

我已经完成了这个测试,知道它是否因SocketException而失败(因为我找不到相关的文档):

    [Test]
    public void DoIt2()
    {
        var byteCount = 0;
        var manualEvents = new[] { new ManualResetEvent(false), new ManualResetEvent(false), new ManualResetEvent(false) };
        var message = new byte[] { 0x01, 0x02, 0x03, 0x04 };
        var socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);

        var callback = new AsyncCallback(ar =>{
            var resetEvent = (ManualResetEvent)ar.AsyncState;
            Interlocked.Add(ref byteCount,  socket.EndSend(ar));
            resetEvent.Set();
        });

        socket.Connect("127.0.0.1", 8000);
        socket.BeginSend(message, 0, message.Length, SocketFlags.None, callback, manualEvents[0]);
        socket.BeginSend(message, 0, message.Length, SocketFlags.None, callback, manualEvents[1]);
        socket.BeginSend(message, 0, message.Length, SocketFlags.None, callback, manualEvents[2]);

        WaitHandle.WaitAll(manualEvents);

        socket.Close();
        Assert.AreEqual(12, byteCount);
    }
它通过了绿色。但是,我需要知道它是否有问题。假设在调用EndSend / EndReceive之前可以多次调用BeginSend / BeginReceive是否安全?我的意思是,是以下的模式吗?

  1. BeginSend
  2. BeginSend
  3. BeginSend
  4. EndSend
  5. EndSend
  6. EndSend
  7. 到货数据顺序怎么样?我知道这是TCP,但我的问题是,鉴于BeginSend是异步的,从1发送的数据是否会在2和3之前到达?

    谢谢。

1 个答案:

答案 0 :(得分:0)

我不知道.NET套接字对象的细节,但一般来说,从不同的线程发送到同一个套接字并不是一个好主意。套接字上发送的数据可以是任何顺序,很可能违反该套接字上的协议。