概述
本篇博客讲述了如何在unity中实现简单的TCP通信功能,包含服务器端与客户端,并配有简易的交互界面,用来显示一些状态信息以及接收和发送的数据。
服务器
首先在服务器端初始化了一个套接字(Socket),并为它绑定上服务器的ip地址与端口号,随后通过创建子线程来监听客户端的连接情况。
//建立tcp通信链接
private void ClickConnect()
{
try
{
int _port = Convert.ToInt32(inputPort); //获取端口号
string _ip = inputIp; //获取ip地址
Debug.Log(" ip 地址是 :" + _ip);
Debug.Log(" 端口号是 :" + _port);
clickConnectBtn = true; //点击了监听按钮
info = "ip地址是 : " + _ip + "端口号是 : " + _port;
//点击开始监听时 在服务端创建一个负责监听IP和端口号的Socket
socketWatch = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
IPAddress ip = IPAddress.Parse(_ip);
IPEndPoint point = new IPEndPoint(ip, _port); //创建对象端口
socketWatch.Bind(point); //绑定端口号
Debug.Log("监听成功!");
info = "监听成功";
socketWatch.Listen(10); //设置监听,最大同时连接10台
//创建监听线程
Thread thread = new Thread(Listen);
thread.IsBackground = true;
thread.Start(socketWatch);
}
catch { }
}
其中监听线程中主要是调用Accept进行阻塞,等待客户端的连接,在连接建立后再创建新的子线程来完成数据的发送和接收功能。
/// <summary>
/// 等待客户端的连接 并且创建与之通信的Socket
/// </summary>
void Listen(object o)
{
try
{
Socket socketWatch = o as Socket;
while (true)
{
socketSend = socketWatch.Accept(); //等待接收客户端连接
Debug.Log(socketSend.RemoteEndPoint.ToString() + ":" + "连接成功!");
info = socketSend.RemoteEndPoint.ToString() + " 连接成功!";
Thread r_thread = new Thread(Received); //开启一个新线程,执行接收消息方法
r_thread.IsBackground = true;
r_thread.Start(socketSend);
Thread s_thread = new Thread(SendMessage); //开启一个新线程,执行发送消息方法
s_thread.IsBackground = true;
s_thread.Start(socketSend);
}
}
catch { }
}
接收消息的子线程将客户端发送来的数据存入字节数组,然后将其转换为字符串并进行显示。
/// <summary>
/// 服务器端不停的接收客户端发来的消息
/// </summary>
void Received(object o)
{
try
{
Socket socketSend = o as Socket;
while (true)
{
byte[] buffer = new byte[1024 * 6]; //客户端连接服务器成功后,服务器接收客户端发送的消息
int len = socketSend.Receive(buffer); //实际接收到的有效字节数
if (len == 0)
{
break;
}
string str = Encoding.UTF8.GetString(buffer, 0, len);
Debug.Log("接收到的消息:" + socketSend.RemoteEndPoint + ":" + str);
recMes = str;
recTimes ++;
info = "接收到一次数据,接收次数为:" + recTimes;
Debug.Log("接收数据次数:" + recTimes);
}
}
catch { }
}
发送消息的子线程实现了从交互界面中获取从键盘输入的发送数据,然后将其转换为字节数组,并通过Send函数发送给客户端。
void SendMessage(object o)
{
try
{
Socket socketSend = o as Socket;
while (true)
{
if(isSendData)
{
isSendData = false;
byte[] sendByte = Encoding.UTF8.GetBytes(inputMessage);
Debug.Log("发送的数据为 :" + inputMessage);
Debug.Log("发送的数据字节长度 :" + sendByte.Length);
socketSend.Send(sendByte);
}
}
}
catch { }
}
最后在程序退出的时候关闭socket并释放相应的资源。
private void OnDisable()
{
Debug.Log("begin OnDisable()");
if (clickConnectBtn)
{
try
{
socketWatch.Shutdown(SocketShutdown.Both); //禁用Socket的发送和接收功能
socketWatch.Close(); //关闭Socket连接并释放所有相关资源
socketSend.Shutdown(SocketShutdown.Both); //禁用Socket的发送和接收功能
socketSend.Close(); //关闭Socket连接并释放所有相关资源
}
catch (Exception e)
{
Debug.Log(e.Message);
}
}
Debug.Log("end OnDisable()");
}
服务器端交互界面中显示了当前TCP通信的一些状态信息,从客户端接收到的数据,要发送给客户端的数据,服务器的ip地址与端口号,监听与数据发送的按钮。
//交互界面
void OnGUI()
{
GUI.color = Color.black;
GUI.Label(new Rect(65, 10, 80, 20), "状态信息");
GUI.Label(new Rect(155, 10, 80, 70), info);
GUI.Label(new Rect(65, 80, 80, 20), "接收到消息:");
GUI.Label(new Rect(155, 80, 80, 20), recMes);
GUI.Label(new Rect(65, 120, 80, 20), "发送的消息:");
inputMessage = GUI.TextField(new Rect(155, 120, 100, 20), inputMessage, 20);
GUI.Label(new Rect(65, 160, 80, 20), "本机ip地址:");
inputIp = GUI.TextField(new Rect(155, 160, 100, 20), inputIp, 20);
GUI.Label(new Rect(65, 200, 80, 20), "本机端口号:");
inputPort = GUI.TextField(new Rect(155, 200, 100, 20), inputPort, 20);
if (GUI.Button(new Rect(65, 240, 60, 20), "开始监听"))
{
ClickConnect();
}
if (GUI.Button(new Rect(65, 280, 60, 20), "发送数据"))
{
isSendData = true;
}
}
客户端
在客户端实现的内容和服务器端类似,首先根据服务器端的ip地址和设置的端口号与服务器建立链接,并创建子线程进行数据的发送和接收。
private void ClickConnect()
{
try
{
int _port = Convert.ToInt32(inputPort); //获取端口号
string _ip = inputIp; //获取ip地址
//创建客户端Socket,获得远程ip和端口号
socketSend = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
IPAddress ip = IPAddress.Parse(_ip);
IPEndPoint point = new IPEndPoint(ip, _port);
socketSend.Connect(point);
Debug.Log("连接成功 , " + " ip = " + ip + " port = " + _port);
staInfo = ip + ":" + _port + " 连接成功";
Thread r_thread = new Thread(Received); //开启新的线程,不停的接收服务器发来的消息
r_thread.IsBackground = true;
r_thread.Start();
Thread s_thread = new Thread(SendMessage); //开启新的线程,不停的给服务器发送消息
s_thread.IsBackground = true;
s_thread.Start();
}
catch (Exception)
{
Debug.Log("IP或者端口号错误......");
staInfo = "IP或者端口号错误......";
}
}
接收消息的子线程和服务器端类似,将接收到的字节数组转换为字符串并进行显示。
void Received()
{
while (true)
{
try
{
byte[] buffer = new byte[1024 * 6];
//实际接收到的有效字节数
int len = socketSend.Receive(buffer);
if (len == 0)
{
break;
}
recMes = Encoding.UTF8.GetString(buffer, 0, len);
Debug.Log("客户端接收到的数据 : " + recMes);
recTimes ++;
staInfo = "接收到一次数据,接收次数为 :" + recTimes;
Debug.Log("接收次数为:" + recTimes);
}
catch { }
}
}
而发送消息的子线程通过点击客户端界面中的按钮进行触发,发送编辑好的信息至服务器。
void SendMessage()
{
try
{
while (true)
{
if (clickSend) //如果点击了发送按钮
{
clickSend = false;
string msg = inputMes;
byte[] buffer = new byte[1024 * 6];
buffer = Encoding.UTF8.GetBytes(msg);
socketSend.Send(buffer);
}
}
}
catch { }
}
最后同样也与服务器端类似,在程序关闭时,禁用Socket的数据发送和接收功能,并释放所有资源。
private void OnDisable()
{
Debug.Log("begin OnDisable()");
if (socketSend.Connected)
{
try
{
socketSend.Shutdown(SocketShutdown.Both); //禁用Socket的发送和接收功能
socketSend.Close(); //关闭Socket连接并释放所有相关资源
}
catch (Exception e)
{
print(e.Message);
}
}
Debug.Log("end OnDisable()");
}
在客户端也创建了一个简易的交互界面,可以用来输入ip地址,建立通信链接按钮与发送数据按钮,显示接收到的数据和一些程序运行的状态信息。
void OnGUI()
{
GUI.color = Color.black;
GUI.Label(new Rect(65, 10, 60, 20), "状态信息");
GUI.Label(new Rect(135, 10, 80, 60), staInfo);
GUI.Label(new Rect(65, 70, 50, 20), "服务器ip地址");
inputIp = GUI.TextField(new Rect(125, 70, 100, 20), inputIp, 20);
GUI.Label(new Rect(65, 110, 50, 20), "服务器端口");
inputPort = GUI.TextField(new Rect(125, 110, 100, 20), inputPort, 20);
GUI.Label(new Rect(65, 150, 80, 20), "接收到消息:");
GUI.Label(new Rect(155, 150, 80, 20), recMes);
GUI.Label(new Rect(65, 190, 80, 20), "发送的消息:");
inputMes = GUI.TextField(new Rect(155, 190, 100, 20), inputMes, 20);
if (GUI.Button(new Rect(65, 230, 60, 20), "开始连接"))
{
ClickConnect();
}
if (GUI.Button(new Rect(65, 270, 60, 20), "发送信息"))
{
clickSend = true;
}
}
需要注意一点的是在发送消息的输入框中键入中文时要将鼠标光标移出unity的场景,不然unity会自动切换到英文输入法,而无法成功输入中文,导致该问题的原因暂时还未找到。
运行结果
服务器端的运行结果如下所示:
客户端的运行结果如下所示:
完整工程
完整的工程存放在百度网盘,大家可以随意取用。
链接:https://pan.baidu.com/s/1KCbXuVhNAJm-YY8EzfH8MA
百度网盘-TCP通信
提取码:e50n