将二进制格式化程序与异步套接字一起使用

时间:2019-05-24 12:15:53

标签: c# sockets asynchronous binaryformatter

我正在尝试利用C#中的异步套接字的优点通过BinaryFormatter进行序列化/反序列化,但是遇到了服务器问题。主要是,我不断收到错误:

No map for object 1953724755

当尝试反序列化网络流时。

我尝试过更改为同步套接字,这似乎可以正常工作,但是在使用异步套接字(使用BeginRecieve和EndRecieve等)时,仍然会出现此错误。

这是我反序列化的代码:

private void BeginDictionaryRecieve(IAsyncResult ar)
{
    Stream socketStream = new NetworkStream(socketState.workSocket);
    IFormatter dataFormatter = new BinaryFormatter();
    socketState.workSocket.EndReceive(ar);
    List<string> rackthatWasClicked = (List<string>) 
        dataFormatter.Deserialize(socketStream);
    this.recievedListForRackDisplay = rackthatWasClicked;
    OnDataRecieved.Invoke();
    socketState.workSocket.BeginReceive(socketState.buffer, 0, StateObject.BufferSize, 0,new AsyncCallback(BeginDictionaryRecieve), socketState);
}

这是序列化代码(同步):

IPAddress ipAdd = IPAddress.Parse("127.0.0.1");
TcpClient myClient = new TcpClient();
myClient.Connect(ipAdd, 8002);
Stream clientStream = myClient.GetStream();
IFormatter f = new BinaryFormatter();
f.Serialize(clientStream, ShelfDataToSend);
clientStream.Dispose();
myClient.Dispose();

我应该注意,我正在使用BinaryFormatter,因为我似乎找不到在套接字上使用常规.Write操作来序列化字符串以外的对象的方法。

1 个答案:

答案 0 :(得分:2)

多亏了汉斯(Hans)的指导,我才找到了一个很好的解决方案。我现在正在接收这样的数据:

try
{
    StateObject state = (StateObject) ar.AsyncState;
    Socket client = state.workSocket;

    int bytesRead = client.EndReceive(ar);

    if (bytesRead > 0)
    {
        stream.Write(state.buffer, 0, bytesRead);
        client.BeginReceive(state.buffer, 0, StateObject.BufferSize, 0, new AsyncCallback(BeginDictionaryRecieve), state);
    }
    else
    {
        if (stream.Length > 1)
        {
            BinaryFormatter formatter = new BinaryFormatter();
            stream.Position = 0;
            this.recievedListForRackDisplay =(List<string>)formatter.Deserialize(stream);
            OnDataRecieved.Invoke();

            client.BeginReceive(state.buffer, 0, StateObject.BufferSize, 0,
                            new AsyncCallback(BeginDictionaryRecieve), state);
        }
    }

}
catch (Exception e)
{
    throw e;
}

还应注意,只有在发送客户端Socket数据后将其丢弃的情况下,此解决方案才有效。