This is a follow on from this question
昨晚我又通过谷歌搜索找到了一个不错的TCP教程,我可以按照它查找IP地址和端口号上的连接,并显示正在发送的数据。
但是,我有一个问题,我的客户端连接一次,我发送一条消息并显示在调试日志中,但是当我停止应用程序并再次运行它时,unity冻结。我不知道为什么会这样。有人能看看这个代码,看看它可能在哪里发生,我能做些什么来修复它吗?
我似乎也会在收到消息后立即启动连接,这是为什么?服务器可以重新连接,但我希望它在拥有连接后保持连接。

public class TCP : MonoBehaviour
{
    string ip_address  = "127.0.0.1";
int port = 22;

Thread listen_thread;
TcpListener tcp_listener;
Thread clientThread;
TcpClient tcp_client;
bool isTrue = true;

// Use this for initialization
void Start ()
{
    IPAddress ip_addy = IPAddress.Parse(ip_address);
    tcp_listener = new TcpListener(ip_addy, port);
    listen_thread = new Thread(new ThreadStart(ListenForClients));
    listen_thread.Start();


    Debug.Log("start thread");

}

private void ListenForClients()
{
    this.tcp_listener.Start();

    while(isTrue == true)
    {
        //blocks until a client has connected to the server
        TcpClient client = this.tcp_listener.AcceptTcpClient();

        //create a thread to handle communication
        //with connected client
        clientThread = new Thread(new ParameterizedThreadStart(HandleClientComm));
        clientThread.Start(client);


        Debug.Log("Got client " + client);

    }
}

private void HandleClientComm(object client)
{
     tcp_client = (TcpClient)client;
    NetworkStream client_stream = tcp_client.GetStream();


    byte[] message = new byte[4096];
    int bytes_read;

    while(isTrue == true)
    {
        bytes_read = 0;

        try
        {
            //blocks until a client sends a message
            bytes_read = client_stream.Read(message, 0, 4096);
            //Debug.Log(message);

        }
        catch (Exception e)
        {
          //a socket error has occured
          Debug.Log(e.Message);
          break;
        }

        if(bytes_read == 0)
        {
            //client has disconnected
            Debug.Log("Disconnected");
            tcp_client.Close();
            break;
        }

        ASCIIEncoding encoder = new ASCIIEncoding();
        Debug.Log(encoder.GetString(message,0,bytes_read));


    }

    if(isTrue == false)
    {
        tcp_client.Close();
        Debug.Log("closing tcp client");
    }

}

void OnApplicationQuit()
{
    try
    {
        tcp_client.Close();
        isTrue = false;
    }
    catch(Exception e)
    {
        Debug.Log(e.Message);
    }
}

}
下面是我的调试日志的屏幕截图,以显示发生了什么:
更新
更新了修复了客户端问题的代码。当我停止unity应用程序并重新启动它时,冻结问题仍然存在。
进一步更新
所以经过进一步的试验,我发现我的项目实际上并没有冻结。当我第一次启动服务器(Unity)应用程序时,一切都正常。但当我关闭它并尝试重新运行服务器时,它会冻结,直到我用客户端连接到它为止。此时服务器正常工作。
所以当我关闭服务器时,我想我不会关闭打开的套接字。我该怎么做?

最佳答案

必须关闭正在侦听的TCP套接字。如果您第一次启动应用程序,TCP套接字将打开。停止应用程序时,TCP套接字仍处于打开状态,并在后台运行。

void OnApplicationQuit()
{
    try
    {
        tcp_client.Close();
        isTrue = false;
    }
    catch(Exception e)
    {
        Debug.Log(e.Message);
    }

    // You must close the tcp listener
    try
    {
        tcp_listener.Stop();
        isTrue = false;
    }
    catch(Exception e)
    {
        Debug.Log(e.Message);
    }
}

08-26 19:10
查看更多