Почему моя форма зависает, когда я вызываю BeginConnect? - PullRequest
1 голос
/ 15 марта 2019

Я использую этот пример из Пример асинхронного клиентского сокета

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

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

public class AsynchronousClient {  
    // The port number for the remote device.  
    private const int port = 11000;  

    // ManualResetEvent instances signal completion.  
    private static ManualResetEvent connectDone =   
        new ManualResetEvent(false);  
    private static ManualResetEvent sendDone =   
        new ManualResetEvent(false);  
    private static ManualResetEvent receiveDone =   
        new ManualResetEvent(false);  

    // The response from the remote device.  
    private static String response = String.Empty;  

    private static void StartClient() {  
        // Connect to a remote device.  
        try {  
            // Establish the remote endpoint for the socket.  
            // The name of the   
            // remote device is "host.contoso.com".  
            IPHostEntry ipHostInfo = Dns.GetHostEntry("host.contoso.com");  
            IPAddress ipAddress = ipHostInfo.AddressList[0];  
            IPEndPoint remoteEP = new IPEndPoint(ipAddress, port);  

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

            // Connect to the remote endpoint.  
            client.BeginConnect( remoteEP,   
                new AsyncCallback(ConnectCallback), client);  
            connectDone.WaitOne();  

            // Send test data to the remote device.  
            Send(client,"This is a test<EOF>");  
            sendDone.WaitOne();  

            // Receive the response from the remote device.  
            Receive(client);  
            receiveDone.WaitOne();  

            // Write the response to the console.  
            Console.WriteLine("Response received : {0}", response);  

            // Release the socket.  
            client.Shutdown(SocketShutdown.Both);  
            client.Close();  

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

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

            // Complete the connection.  
            client.EndConnect(ar);  

            Console.WriteLine("Socket connected to {0}",  
                client.RemoteEndPoint.ToString());  

            // Signal that the connection has been made.  
            connectDone.Set();  
        } catch (Exception e) {  
            Console.WriteLine(e.ToString());  
        }  
    }  

    private static void Receive(Socket client) {  
        try {  
            // Create the state object.  
            StateObject state = new StateObject();  
            state.workSocket = client;  

            // Begin receiving the data from the remote device.  
            client.BeginReceive( state.buffer, 0, StateObject.BufferSize, 0,  
                new AsyncCallback(ReceiveCallback), state);  
        } catch (Exception e) {  
            Console.WriteLine(e.ToString());  
        }  
    }  

    private static void ReceiveCallback( IAsyncResult ar ) {  
        try {  
            // Retrieve the state object and the client socket   
            // from the asynchronous state object.  
            StateObject state = (StateObject) ar.AsyncState;  
            Socket client = state.workSocket;  

            // Read data from the remote device.  
            int bytesRead = client.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));  

                // Get the rest of the data.  
                client.BeginReceive(state.buffer,0,StateObject.BufferSize,0,  
                    new AsyncCallback(ReceiveCallback), state);  
            } else {  
                // All the data has arrived; put it in response.  
                if (state.sb.Length > 1) {  
                    response = state.sb.ToString();  
                }  
                // Signal that all bytes have been received.  
                receiveDone.Set();  
            }  
        } catch (Exception e) {  
            Console.WriteLine(e.ToString());  
        }  
    }  

    private static void Send(Socket client, 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.  
        client.BeginSend(byteData, 0, byteData.Length, 0,  
            new AsyncCallback(SendCallback), client);  
    }  

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

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

            // Signal that all bytes have been sent.  
            sendDone.Set();  
        } catch (Exception e) {  
            Console.WriteLine(e.ToString());  
        }  
    }  

    public static int Main(String[] args) {  
        StartClient();  
        return 0;  
    }  
}  

Я сейчас занимаюсь программированием сетевой части своего приложения, сейчас я просто тестирую, но у меня странная проблема. Мой пользовательский интерфейс зависает, когда я вызываю метод StartClient. Этот пример находится в отдельном классе, и просто вызовите его в моей основной форме с помощью: AsynchronousClient.Startclient();

Я должен отметить, что сейчас сервер не слушает, но почему форма зависает, в конце концов, я не использую синхронный пример. Я читал в Интернете, что это может быть потому, что поток формы должен быть запущен из основного потока, но я ничего не изменил относительно этого. По моему мнению, форма должна просто продолжать функционировать, даже если она не может подключиться.

Как мне этого добиться? Я могу дать вам свой код, но я думаю, что он не очень актуален. Как я уже говорил, я вызываю метод StartClient из класса AsynchronousClient, он статический, а затем вызываю его где-нибудь в моем основном классе. (в кнопке прямо сейчас).

Не имеет значения, я думаю, но я знаю, что пакеты отправлены, я проверил это с помощью wireshark.

1 Ответ

0 голосов
/ 15 марта 2019

Согласно моему комментарию, AsynchronousClient.Startclient() не является асинхронным, поскольку для блокировки используется ManualResetEvent.

Я лично рекомендую не использовать Socket напрямую. Это действительно зависит от того, что вы делаете, но есть лучшие API для удаленного общения.

HTTP, Json-RPC, gRPC, ZeroMQ и т. Д. Зависит от вашего варианта использования. Использование необработанных TCP-сокетов почти всегда избыточно.

В любом случае, чтобы сделать метод действительно асинхронным, вам нужно избавиться от вызовов WaitOne.
Я считаю, что причина вызовов WaitOne в первую очередь в том, чтобы сделать код более читабельным, но теоретически вы можете выполнить логику «Отправка» в обратном вызове «Соединение» и логику «Получение» в «Отправка» обратного вызова.

Как это

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

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

public class AsynchronousClient {  
    // The port number for the remote device.  
    private const int port = 11000;  

    // ManualResetEvent instances signal completion.  
    private static ManualResetEvent connectDone =   
        new ManualResetEvent(false);  
    private static ManualResetEvent sendDone =   
        new ManualResetEvent(false);  
    private static ManualResetEvent receiveDone =   
        new ManualResetEvent(false);  

    // The response from the remote device.  
    private static String response = String.Empty;  

    private static void StartClient() {  
        // Connect to a remote device.  
        try {  
            // Establish the remote endpoint for the socket.  
            // The name of the   
            // remote device is "host.contoso.com".  
            IPHostEntry ipHostInfo = Dns.GetHostEntry("host.contoso.com");  
            IPAddress ipAddress = ipHostInfo.AddressList[0];  
            IPEndPoint remoteEP = new IPEndPoint(ipAddress, port);  

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

            // Connect to the remote endpoint.  
            client.BeginConnect( remoteEP,   
                new AsyncCallback(ConnectCallback), client);  


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

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

            // Complete the connection.  
            client.EndConnect(ar);  

            Console.WriteLine("Socket connected to {0}",  
                client.RemoteEndPoint.ToString());  

            // we do it here now...
            Send(client,"This is a test<EOF>");  
        } catch (Exception e) {  
            Console.WriteLine(e.ToString());  
        }  
    }  

    private static void Receive(Socket client) {  
        try {  
            // Create the state object.  
            StateObject state = new StateObject();  
            state.workSocket = client;  

            // Begin receiving the data from the remote device.  
            client.BeginReceive( state.buffer, 0, StateObject.BufferSize, 0,  
                new AsyncCallback(ReceiveCallback), state);  
        } catch (Exception e) {  
            Console.WriteLine(e.ToString());  
        }  
    }  

    private static void ReceiveCallback( IAsyncResult ar ) {  
        try {  
            // Retrieve the state object and the client socket   
            // from the asynchronous state object.  
            StateObject state = (StateObject) ar.AsyncState;  
            Socket client = state.workSocket;  

            // Read data from the remote device.  
            int bytesRead = client.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));  

                // Get the rest of the data.  
                client.BeginReceive(state.buffer,0,StateObject.BufferSize,0,  
                    new AsyncCallback(ReceiveCallback), state);  
            } else {  
                // All the data has arrived; put it in response.  
                if (state.sb.Length > 1) {  
                    response = state.sb.ToString();  
                }  

                // after all is done, we shut down the socket
                client.Shutdown(SocketShutdown.Both);  
                client.Close();  
            }  
        } catch (Exception e) {  
            Console.WriteLine(e.ToString());  
        }  
    }  

    private static void Send(Socket client, 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.  
        client.BeginSend(byteData, 0, byteData.Length, 0,  
            new AsyncCallback(SendCallback), client);  
    }  

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

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

            // We do it here now...
            Receive(client); 
        } catch (Exception e) {  
            Console.WriteLine(e.ToString());  
        }  
    }  

    public static int Main(String[] args) {  
        StartClient();  
        return 0;  
    }  
} 

Это должно работать. При этом StartClient не блокируется. Возвращается сразу после вызова BeginConnect.

Однако это очень старый и грязный API, поэтому я подумал бы дважды, прежде чем использовать нативные Socket s.

Вы можете лучше объяснить свою цель, а я постараюсь предложить что-то лучше.

...