.net - 异步套接字服务器C#,如何从AsyncCallback获取数据

标签 .net sockets asynchronous tcp c#-2.0

我一直在努力学习使用异步套接字,我确信我的问题非常简单,但我一直很难找到解决方案。我将从 MSDN 上为 Asynchronous Server 提供的代码开始和 Asynchronous Client .我正在关注代码的工作原理,除了一个大问题:如何获取客户端发送回程序主函数的字符串?我在 ReadCallback() 函数中看到了字符串完全形成的位置。

问题是如何将字符串内容返回给Main()?我发表了一些评论,试图阐明我想要达到的目标。预先感谢您的帮助。

异步服务器:

using System;  
using System.Net;  
using System.Net.Sockets;  
using System.Text;  
using System.Threading;  

// State object for reading client data asynchronously  
public class StateObject {  
    // Client  socket.  
    public Socket workSocket = null;  
    // Size of receive buffer.  
    public const int BufferSize = 1024;  
    // Receive buffer.  
    public byte[] buffer = new byte[BufferSize];  
// Received data string.  
    public StringBuilder sb = new StringBuilder();    
}  

public class AsynchronousSocketListener {  
    // Thread signal.  
    public static ManualResetEvent allDone = new ManualResetEvent(false);  

    public AsynchronousSocketListener() {  
    }  

    public static void StartListening() {  
        // Data buffer for incoming data.  
        byte[] bytes = new Byte[1024];  

        // Establish the local endpoint for the socket.  
        // The DNS name of the computer  
        // running the listener is "host.contoso.com".  
        IPHostEntry ipHostInfo = Dns.Resolve(Dns.GetHostName());  
        IPAddress ipAddress = ipHostInfo.AddressList[0];  
        IPEndPoint localEndPoint = new IPEndPoint(ipAddress, 11000);  

        // Create a TCP/IP socket.  
        Socket listener = new Socket(AddressFamily.InterNetwork,  
            SocketType.Stream, ProtocolType.Tcp );  

        // Bind the socket to the local endpoint and listen for incoming connections.  
        try {  
            listener.Bind(localEndPoint);  
            listener.Listen(100);  

            while (true) {  
                // Set the event to nonsignaled state.  
                allDone.Reset();  

                // Start an asynchronous socket to listen for connections.  
                Console.WriteLine("Waiting for a connection...");  
                listener.BeginAccept(   
                    new AsyncCallback(AcceptCallback),  
                    listener );  

                // Wait until a connection is made before continuing.  
                allDone.WaitOne();  
            }  

        } catch (Exception e) {  
            Console.WriteLine(e.ToString());  
        }  

        Console.WriteLine("\nPress ENTER to continue...");  
        Console.Read();  

    }  

    public static void AcceptCallback(IAsyncResult ar) {  
        // Signal the main thread to continue.  
        allDone.Set();  

        // Get the socket that handles the client request.  
        Socket listener = (Socket) ar.AsyncState;  
        Socket handler = listener.EndAccept(ar);  

        // Create the state object.  
        StateObject state = new StateObject();  
        state.workSocket = handler;  
        handler.BeginReceive( state.buffer, 0, StateObject.BufferSize, 0,  
            new AsyncCallback(ReadCallback), state);  
    }  

    public static void ReadCallback(IAsyncResult ar) {  
        String content = String.Empty;  

        // Retrieve the state object and the handler socket  
        // from the asynchronous state object.  
        StateObject state = (StateObject) ar.AsyncState;  
        Socket handler = state.workSocket;  

        // Read data from the client socket.   
        int bytesRead = handler.EndReceive(ar);  

        if (bytesRead > 0) {  
            // There  might be more data, so store the data received so far.  
            state.sb.Append(Encoding.ASCII.GetString(  
                state.buffer,0,bytesRead));  

            // Check for end-of-file tag. If it is not there, read   
            // more data.  
            content = state.sb.ToString();  
            if (content.IndexOf("<EOF>") > -1) {  
                // All the data has been read from the   
                // client. Display it on the console.  
                Console.WriteLine("Read {0} bytes from socket. \n Data : {1}",  
                    content.Length, content );  
                // Echo the data back to the client. 
                // AT THIS POINT content SHOULD HAVE THE STRING SENT 
                Send(handler, content);  
            } else {  
                // Not all data received. Get more.  
                handler.BeginReceive(state.buffer, 0, StateObject.BufferSize, 0,  
                new AsyncCallback(ReadCallback), state);  
            }  
        }  
    }  

    private static void Send(Socket handler, String data) {  
        // Convert the string data to byte data using ASCII encoding.  
        byte[] byteData = Encoding.ASCII.GetBytes(data);  

        // Begin sending the data to the remote device.  
        handler.BeginSend(byteData, 0, byteData.Length, 0,  
            new AsyncCallback(SendCallback), handler);  
    }  

    private static void SendCallback(IAsyncResult ar) {  
        try {  
            // Retrieve the socket from the state object.  
            Socket handler = (Socket) ar.AsyncState;  

            // Complete sending the data to the remote device.  
            int bytesSent = handler.EndSend(ar);  
            Console.WriteLine("Sent {0} bytes to client.", bytesSent);  

            handler.Shutdown(SocketShutdown.Both);  
            handler.Close();  

        } catch (Exception e) {  
            Console.WriteLine(e.ToString());  
        }  
    }  

    public static int Main(String[] args) {  
        StartListening();  
        // I want to make string here that has the result from ReadCallback so I can then perform other logic on the string.
        return 0;  
    }  
} 

最佳答案

您完全按照示例进行了操作。你只需要等待输入。在返回语句之前放置一个 Console.ReadLine();:

public static int Main(String[] args) 
{  
    StartListening();  
    // I want to make string here that has the result from ReadCallback so I can then perform other logic on the string.

    Console.ReadLine();
    return 0;  
} 

如果你想把输入打印出来。在 ReadCallback 方法中解码接收到的字节后,只需使用 Console.WriteLine():

if (bytesRead > 0) {  

    // There  might be more data, so store the data received so far.  
    state.sb.Append(Encoding.ASCII.GetString(  
        state.buffer,0,bytesRead));  

    // Check for end-of-file tag. If it is not there, read   
    // more data.  
    content = state.sb.ToString();  

    // print the stuff to the console
    Console.WriteLine(content);

如果你想拥有内容并对其进行一些操作,你可以将它声明为静态类变量,你将能够在 main 方法中访问它:

public class AsynchronousSocketListener 
{
    public static string content = "";


public static int Main(String[] args) 
{  
    StartListening();  
    // I want to make string here that has the result from ReadCallback so I can then perform other logic on the string.

    Console.ReadLine();

    // wait until the message has arrive and press enter to jump to this line
    Console.WriteLine("Now I can do what ever I want with the incoming message: " + content);
    // just to be able to read it one more time before the program ends ;)
    Console.ReadLine();
    return 0;  
} 

当然,您必须从 ReadCallback 中删除声明。

您必须知道它是异步的,因此您必须等待消息到达。但是如果你在那里打印它,你可以在控制台中看到它。剩下的由你决定

关于.net - 异步套接字服务器C#,如何从AsyncCallback获取数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42169574/

相关文章:

.net - GetIpAddrTable 的任何替代方案?

c - IPv6结构定义错误

c# - 从 C# 调用和等待 F# 异步方法

ios - 如何在不破坏内置抽象的情况下将 websocket 请求路由到 SailsJS

linux - 在 AF_PACKET 套接字上发送数据

javascript - 如何一一执行三个异步函数

javascript - Node.js - async.queue() - 如何结束任务以允许更多任务运行?

c# - 如何优雅地处理 winforms 应用程序中的休眠/ sleep 模式?

.net - 如何使用Grails调用WCF .net Web服务

.net - 当 F# 中的类型提供程序更改时会发生什么?