目录
一、客户端代码
data:image/s3,"s3://crabby-images/cfe28/cfe2855c0d4b2b1e3c523c3443221a07a28f909c" alt=""
data:image/s3,"s3://crabby-images/a1a2b/a1a2b0d00381edf6ad3ca671944efcdba2456cc5" alt=""
data:image/s3,"s3://crabby-images/8265f/8265f216eec5c0d2221527d263fbf2bc64ffcba8" alt=""
连接本地服务器127.0.0.1:2012端口(如何创本地服务器,放最后说),连接成功后会回调
data:image/s3,"s3://crabby-images/61102/6110210c08fa96ed037dbd51714eb88ec423dfe2" alt=""
data:image/s3,"s3://crabby-images/8d554/8d554912cf9bbae257aff1a363f8e095c14bda94" alt=""
协议号Connect是101,其他如下,将这个Connect协议号和数据(空的字节流)放入一个mEvents队列,等待Update执行(目的应该是回到主线程才执行回调,避免一些问题)
data:image/s3,"s3://crabby-images/a1464/a1464754bc1ddd0c9b581c04b56ff0142d9fda63" alt=""
data:image/s3,"s3://crabby-images/0c944/0c94438f59a3d94d153e908e7a5e4c694cd2e7ed" alt=""
这里Update方法就是Monobehaviour的生命周期Update,它简单地遍历队列逐个出队派发事件"DISPATCH_MESSAGE"去到如下
data:image/s3,"s3://crabby-images/88af5/88af5b6c91d5e3a8f8b4fec42b2f4d8ac16ad47f" alt=""
再回到调用lua的Network.OnSocket方法并且将 buffer.Key(协议号 101) 和 buffer.Value(空) 数据传递。
data:image/s3,"s3://crabby-images/92425/924254984d0c10fb9ddde04544114a2764d43fda" alt=""
Network.OnSocket继续派发协议号作为事件名,执行OnConnect方法,至此完成一整套连接流程
data:image/s3,"s3://crabby-images/f7bb8/f7bb86e6a0da96f254613c683917e37c3b711d66" alt=""
其他相关的事件可以查如下找到相关的代码
data:image/s3,"s3://crabby-images/9df6b/9df6be9cb69055ad203f95c6d873805858923680" alt=""
由于我们lua侧仅写了1个消息协议号'104' 所以本地服务器代码也是要用104作为协议号传递,不然接收消息检测不到是104就无法正常通过消息派发并执行到对应的lua消息回调代码。
data:image/s3,"s3://crabby-images/5231f/5231f7ceccb1a3be64ad0b753b65e81be5cbe5b2" alt=""
接收消息回调
data:image/s3,"s3://crabby-images/a36b5/a36b5d376b17ebb650403ecbf7acae4b39580a58" alt=""
TestProtoType有很多种传递消息的方式,项目默认使用ProtocalType.BINARY
data:image/s3,"s3://crabby-images/da795/da7950e275eee34cee5fffd924874ff8072ef8c5" alt=""
data:image/s3,"s3://crabby-images/b6c30/b6c302cbafcdb8d9258e3b6dc99bc3052ae82aef" alt=""
data:image/s3,"s3://crabby-images/5be04/5be04963274636964293fc1697188a9ead184e67" alt=""
data:image/s3,"s3://crabby-images/d298e/d298e34e5e2d2bfff2bd5bd10fef15e3fca958c2" alt=""
data:image/s3,"s3://crabby-images/c68cb/c68cbdadf14e3616d4fe17abb1db5dd6b8830b02" alt=""
好绕,最终是到了上面的WriteMessage方法,发送是以"消息长度"+"消息内容"为一条字节流传递到服务器,执行完后回调OnWrite方法
data:image/s3,"s3://crabby-images/81703/81703679e7c22f6797a40f547c91d87ac070aac1" alt=""
二、本地服务器代码
data:image/s3,"s3://crabby-images/182b6/182b659776fe36f41ccfda5fb400635e83db256b" alt=""
cs
using System;
using System.Net;
using System.Net.Sockets;
namespace SimpleNet
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine("Hello World!");
//Socket tcp
Socket listenfd = new Socket(AddressFamily.InterNetwork,
SocketType.Stream, ProtocolType.Tcp);
//bind
IPAddress ipAdr = IPAddress.Parse("127.0.0.1");
IPEndPoint ipEp = new IPEndPoint(ipAdr, 2012);
listenfd.Bind(ipEp);
//listen
listenfd.Listen(0);
Console.WriteLine("启动服务器成功");
while (true)
{
//Accept
Socket connfd = listenfd.Accept();
Console.WriteLine("服务器Accept");
//Recv 测试
byte[] readBuff = new byte[100];
int count = connfd.Receive(readBuff);
//steam
string showStr = "";
for(int i = 0; i< count; i++)
{
int b = (int)readBuff[i];
showStr += b.ToString() + " ";
}
Console.WriteLine("字节流:" + showStr);
//解析
Int16 msgLen = BitConverter.ToInt16(readBuff, 0);
Int16 protocal = BitConverter.ToInt16(readBuff, 2);
Int16 strLen = BitConverter.ToInt16(readBuff, 5);//具体还要看字节流输出来读取第几个字节是长度,若读错了下一行会报错
string str = System.Text.Encoding.UTF8.GetString(readBuff, 6, strLen);
Console.WriteLine("消息长度:" + msgLen);
Console.WriteLine("协议号:" + protocal);
Console.WriteLine("字符串:" + str);
//send 原样返回
byte[] writeBuff = new byte[count];
Array.Copy(readBuff, writeBuff, count);
connfd.Send(writeBuff);
}
}
}
}
data:image/s3,"s3://crabby-images/8042c/8042ce18a739dfcff928d30fe49660da9f500084" alt=""
data:image/s3,"s3://crabby-images/3986f/3986f65fc72d3f43d57f89368e1a35728410317a" alt=""
客户端接收到服务器消息的打印,服务器是直接将客户端的消息拷贝了一份再发送给客户端。
跑代码时发现有些lua报错:FindChild相关不存在
--初始化面板--
function MessagePanel.InitPanel()
--this.btnClose = transform:FindChild("Button").gameObject; --bug代码 FindChild是不存在的接口...
this.btnClose = transform:Find("Button").gameObject;
end
三、解决服务器无法多次接收客户端消息问题
测试发现上面的服务器代码不支持多次发送和接收消息,可以修改为如下服务器代码解决:
cs
using System;
using System.Net;
using System.Net.Sockets;
using System.Threading.Tasks;
namespace SimpleNet
{
class Program
{
static async Task Main(string[] args)
{
SimpleTcpServer server = new SimpleTcpServer("127.0.0.1", 2012);
await server.StartAsync();
}
}
}
cs
using System;
using System.Net;
using System.Net.Sockets;
using System.Text;
using System.Threading.Tasks;
namespace SimpleNet
{
public class SimpleTcpServer
{
private TcpListener _listener;
private string _ip;
private int _port;
public SimpleTcpServer(string ip, int port)
{
_ip = ip;
_port = port;
}
public async Task StartAsync()
{
_listener = new TcpListener(IPAddress.Parse(_ip), _port);
_listener.Start();
Console.WriteLine($"Server started on port {_port}.");
while (true)
{
TcpClient client = await _listener.AcceptTcpClientAsync();
_ = HandleClientAsync(client);
}
}
private async Task HandleClientAsync(TcpClient client)
{
NetworkStream stream = client.GetStream();
byte[] readBuff = new byte[1024];
while (true)
{
try
{
int count = await stream.ReadAsync(readBuff, 0, readBuff.Length);
if (count == 0) // Client disconnected
{
break;
}
//steam
string showStr = "";
for (int i = 0; i < count; i++)
{
int b = (int)readBuff[i];
showStr += b.ToString() + " ";
}
Console.WriteLine("字节流:" + showStr);
//解析
Int16 msgLen = BitConverter.ToInt16(readBuff, 0);
Int16 protocal = BitConverter.ToInt16(readBuff, 2);
Int16 strLen = BitConverter.ToInt16(readBuff, 5);
string str = System.Text.Encoding.UTF8.GetString(readBuff, 7, strLen);
Console.WriteLine("消息长度:" + msgLen);
Console.WriteLine("协议号:" + protocal);
Console.WriteLine("字符串:" + str);
//send 原样返回
byte[] writeBuff = new byte[count];
Array.Copy(readBuff, writeBuff, count);
await stream.WriteAsync(writeBuff, 0, writeBuff.Length);
}
catch (Exception ex)
{
Console.WriteLine("Error: " + ex.Message);
break;
}
}
client.Close();
}
public void Stop()
{
_listener.Stop();
}
}
}