Unity使用webSocket与服务器通信(二)——C#服务器端使用Fleck时的简单服用方法
创始人
2024-05-31 06:22:48
0

C#服务端用到Fleck包,它包含哪些可用的回调函数,有哪些常用的api方法?

演示:服务端收到Unity用户发来的信息
请添加图片描述

1、Fleck服务器提供哪些回调函数

Fleck提供的回调函数有下面几种:

//用户连入服务器时...
Action OnOpen { get; set; }//用户与服务器断开连接时...
Action OnClose { get; set; }//收到字符串消息时...
Action OnMessage { get; set; }//收到二进制数据时...
Action OnBinary { get; set; }//收到别人发来的ping信息时...
Action OnPing { get; set; }//收到别人发来的pong信息时...
Action OnPong { get; set; }//出错的时候调用...[?谁出错,服务器出错还是连接出错,出的是什么错?]
Action OnError { get; set; }

2 、服务器提供的其它API

其它常用的api主要有:

//发送字符串
Task Send(string message);//发送字节码(二进制数据)
// 1Byte = 8 bits
// 1KB = 1024 Bytes
// 1MB = 1024KB
Task Send(byte[] message);//发送一个ping信息
Task SendPing(byte[] message);//发送一个pong信息
Task SendPong(byte[] message);//关闭连接
void Close();//关闭连接?关闭连接池中指定序号的连接?
void Close(int code);
  • ping pong的作用是啥?
    WebSocket为了保持客户端、服务端的实时双向通信,需要确保客户端、服务端之间的TCP通道保持连接没有断开。然而,对于长时间没有数据往来的连接,如果依旧长时间保持着,可能会浪费包括的连接资源。但不排除有些场景,客户端、服务端虽然长时间没有数据往来,但仍需要保持连接。这个时候,可以采用心跳来实现。
    发送方->接收方:ping;
    接收方->发送方:pong;

C# 在WinForm里使用Fleck作为服务器的简单示例代码:

 private void button1_Click(object sender, EventArgs e)
{if (hasStartedServer)//不重复启动服务器实例return;var server = new WebSocketServer("ws://192.168.0.137:8081");  //ws://localhost:8081    ws://127.0.0.0:8181server.Start(socket =>{//连上时...socket.OnOpen = () =>{Debug.WriteLine($"有新用户连入:{socket.ConnectionInfo.ClientIpAddress}");};//断开时...socket.OnClose = () =>{Debug.WriteLine($"用户断开连接:{socket.ConnectionInfo.ClientIpAddress}");UserSockets.First(x => x.socket.ConnectionInfo.Id == socket.ConnectionInfo.Id).connected = false;};//收到string信息时...socket.OnMessage = message =>{socket.Send($"服务器收到消息 : {DateTime.Now.ToString()}");Debug.WriteLine($"收到一条消息,来自:{socket.ConnectionInfo.ClientIpAddress}");Debug.WriteLine(message);var cmd  = message.Split("#")[0];if (cmd == "name"){var userID = message.Split("#")[1];Debug.WriteLine($"收到一条消息,来自用户:{userID}  连接id:{socket.ConnectionInfo.Id}");UserSocket user = new UserSocket(){userID = userID,socket = socket,connected = true};UserSockets.Add(user);}};//收到二进制信息时...socket.OnBinary = bytes =>{var userName = UserSockets.First(x => x.socket.ConnectionInfo.Id == socket.ConnectionInfo.Id).userID;Debug.WriteLine($"收到二进制数据,长度为{bytes.Length}Bytes,来自ip:{socket.ConnectionInfo.ClientIpAddress},userID ={userName}");};});Debug.WriteLine("服务器已经启动!");hasStartedServer = true;
}

3、服务器与客户端建立的连接包含哪些信息

  public interface IWebSocketConnectionInfo{string SubProtocol { get; }string Origin { get; }string Host { get; }string Path { get; }string ClientIpAddress { get; }int ClientPort { get; }IDictionary Cookies { get; }IDictionary Headers { get; }Guid Id { get; }string NegotiatedSubProtocol { get; }}

每次建立的连接,Id号是唯一的。

4、服务器接收数据时是否会粘包?

经测试,同一连接连续向服务器发送数据时,每次OnBinary收到的消息是完整的。
还没瞻仰源码,后面有空看看,server端TCP接收数据时,应该是做了包的合并处理,接到一个整坨数据,才调用的OnBinary。
在这里插入图片描述

请添加图片描述

  • Unity用户端代码
    连发50笔数据,每次发送1M bytes
 int i = 0;while (i < 50)  //连发50笔数据,每次发送1M bytes{var bytesArray = new byte[1048576];  // 1024 * 1024 = 1048576await websocket.Send(bytesArray);    //或者直接await,测试结果一样Debug.Log($"发送数据-{i}");i++;}
  • C#服务端代码
    每当收到数据时,把收到的数据长度打印出来
socket.OnBinary = bytes =>{Debug.WriteLine($"收到二进制数据,长度为{bytes.Length}Bytes,来自{socket.ConnectionInfo.ClientIpAddress}");};

5、同一个ip上有两个应用同时发来信息,如何区分这些连接属于哪个用户?

简要思路:每个用户端启动的时候,需要用户名登录,建立连接时,告诉服务器这个socket是哪个userID的,凡是该用户建立连接,后台都把该连接绑定到user ID。

  • 1、每次建立连接,connection的id是唯一的
    在这里插入图片描述

  • 2、服务器端维护一个【连接列表】:(string userID - > webSocket sockt),如下所示:

public class UserSocket
{/// /// 用户ID/// public string userID;/// /// socket对象/// public IWebSocketConnection socket;/// /// 状态:true-可用状态  false-断开状态/// public bool connected;
}
userIDsocketconnected
guestsocket1true
user1socket2true
user2socket3false
userNsocketMtrue

客户端建立连接的时候,发送信息给服务器,告诉服务器该连接对应那个用户名。
服务器收到信息后,及时更新【连接列表】。

  • 用户端在建立连接的时候,发送一个string命令,告诉服务器这个连接后面是哪个用户(user ID)
websocket.OnOpen += () =>
{Debug.Log("连接成功!!");   websocket.SendText($"name#user001"); //发送用户id
};
  • 服务器收到信息的处理
 //收到二进制信息
socket.OnBinary = bytes =>
{var userName = UserSockets.First(x => x.socket.ConnectionInfo.Id == socket.ConnectionInfo.Id).userID;Debug.WriteLine($"收到二进制数据,长度为{bytes.Length}Bytes,来自ip:{socket.ConnectionInfo.ClientIpAddress},userID ={userName}");
};

收到信息,且识别了是哪个userID发来的
请添加图片描述

6、本文测试环境

Win10 + Unity2021.3.18 + VS2019(.NET 5.0)

相关内容

热门资讯

【观点撷英】加强学术研究力量推... 首届产业工人队伍建设改革学术研讨会近日在南开大学举行。与会专家学者从跨学科、跨领域视角,围绕产业工人...
助力银发族畅享“诗和远方” “丽江看雪山,腾冲泡温泉,保山喝咖啡,住得舒适,吃得健康有特色,行程安排得适中,管家也很贴心。”62...
首个全国性“医保+商保”清分结...   本报讯(记者 柴嵘)“以前商保报销,单据得攒一沓,报销的钱到账少则三五天,多则要等十多天。现在医...
风语筑李晖:好玩的时候才刚刚开... 7月5日,上海市奉贤区“上海之鱼”热闹非凡。刚刚放暑假的孩子、《三体》迷、科幻迷纷纷来到最新开幕的在...
偶像的真正含义是什么? 偶像的真正含义是什么?偶像,难道就是这个人很帅,很漂亮,就是偶像了吗?可是,偶像的真正含义到底是什么...
助力城市副中心文旅高质量发展   本报讯(记者 陈施君)为助力副中心文旅发展,更好地履行民主监督职能,根据民主监督协商议题计划安排...
网贷利息计算器 网贷利息计算器网贷利息计算器贷款4万到账4万,分48期,每期1566.15.年利息百分之几
职工的“钱袋子”鼓起来了 近日,内蒙古森工集团甘河森工公司二楼会议室,职工们一边交流着自己对黑木耳种植的见解、打算,一边等待领...
铡刀药匾间的匠心传承 走进江西樟树中医药职业学院实训室,热闹非凡,空气中弥漫着独特的药香。操作台上,整齐摆放着的黄芪、白芍...
单纯扩散,易化扩散,主动转运和... 单纯扩散,易化扩散,主动转运和包吞包吐有何主要区别?单纯扩散,一般是脂溶性小分子,由高浓度到低浓度,...