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);
}
}