HaoYueNet/NetLib/HaoYueNet.ClientNetwork/NetworkHelperCore.cs

383 lines
13 KiB
C#
Raw Permalink Normal View History

//using HunterProtobufCore;
2023-06-02 18:24:54 +08:00
using System.Net;
2021-12-10 23:38:40 +08:00
using System.Net.Sockets;
using static HaoYueNet.ClientNetwork.BaseData;
2021-12-10 23:38:40 +08:00
namespace HaoYueNet.ClientNetwork
{
public class NetworkHelperCore
{
private Socket client;
////响应倒计时计数最大值
//private static int MaxRevIndexNum = 6;
////发送倒计时计数最大值
//private static int MaxSendIndexNum = 3;
//响应倒计时计数最大值
private static int MaxRevIndexNum = 50;
//发送倒计时计数最大值
private static int MaxSendIndexNum = 3;
//响应倒计时计数
private static int RevIndex=0;
//发送倒计时计数
private static int SendIndex=0;
//计时器间隔
private static int TimerInterval = 3000;
private System.Timers.Timer _heartTimer;
2023-12-14 18:15:04 +08:00
public static string LastConnectIP;
public static int LastConnectPort;
2023-12-25 17:37:33 +08:00
public bool bDetailedLog = false;
2021-12-10 23:38:40 +08:00
2023-12-25 17:37:33 +08:00
public bool Init(string IP, int port,bool isHadDetailedLog = true, bool bBindReuseAddress = false,int bBindport = 0)
2023-12-14 18:15:04 +08:00
{
2021-12-10 23:38:40 +08:00
LogOut("==>初始化网络核心");
2023-12-25 17:37:33 +08:00
bDetailedLog = isHadDetailedLog;
2021-12-10 23:38:40 +08:00
RevIndex = MaxRevIndexNum;
SendIndex = MaxSendIndexNum;
client = new Socket(SocketType.Stream, ProtocolType.Tcp);
2023-06-15 16:52:33 +08:00
if (bBindReuseAddress)
{
client.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.ReuseAddress, true);
IPEndPoint ipe = new IPEndPoint(IPAddress.Any, Convert.ToInt32(bBindport));
client.Bind(ipe);
}
2023-12-14 18:15:04 +08:00
LastConnectIP = IP;
LastConnectPort = port;
return Connect(IP, port);
2023-06-15 16:52:33 +08:00
}
2021-12-10 23:38:40 +08:00
2023-12-14 18:15:04 +08:00
bool Connect(string IP, int port)
2023-06-15 16:52:33 +08:00
{
2021-12-10 23:38:40 +08:00
//带回调的
try
{
2023-12-25 17:37:33 +08:00
if(bDetailedLog)
LogOut("连接到远程IP " + IP + ":" + port);
else
LogOut("连接到远程服务");
2021-12-10 23:38:40 +08:00
client.Connect(IP, port);
Thread thread = new Thread(Recive);
thread.IsBackground = true;
thread.Start(client);
2023-06-15 16:52:33 +08:00
int localport = ((IPEndPoint)client.LocalEndPoint).Port;
2023-12-25 17:37:33 +08:00
if (bDetailedLog)
LogOut($"连接成功!连接到远程IP->{IP}:{port} | 本地端口->{localport}");
else
LogOut("连接成功!");
2021-12-10 23:38:40 +08:00
2023-06-15 16:52:33 +08:00
if (_heartTimer == null)
{
_heartTimer = new System.Timers.Timer();
}
2021-12-10 23:38:40 +08:00
_heartTimer.Interval = TimerInterval;
_heartTimer.Elapsed += CheckUpdatetimer_Elapsed;
_heartTimer.AutoReset = true;
_heartTimer.Enabled = true;
2023-12-25 17:37:33 +08:00
if (bDetailedLog)
LogOut("开启心跳包检测");
2021-12-10 23:38:40 +08:00
2023-07-06 14:47:20 +08:00
OnConnected?.Invoke(true);
2023-06-15 16:52:33 +08:00
return true;
2021-12-10 23:38:40 +08:00
}
2023-06-15 16:52:33 +08:00
catch (Exception ex)
2021-12-10 23:38:40 +08:00
{
2023-12-25 17:37:33 +08:00
if (bDetailedLog)
LogOut("连接失败:" + ex.ToString());
else
LogOut("连接失败");
2023-07-06 14:47:20 +08:00
OnConnected?.Invoke(false);
2023-06-15 16:52:33 +08:00
return false;
2021-12-10 23:38:40 +08:00
}
}
~NetworkHelperCore()
{
client.Close();
}
private void SendToSocket(byte[] data)
{
2024-01-08 16:19:09 +08:00
//已拼接包长度,这里不再需要拼接长度
//data = SendDataWithHead(data);
2021-12-10 23:38:40 +08:00
try
{
SendWithIndex(data);
}
catch (Exception ex)
{
//连接断开
OnCloseReady();
return;
}
2023-06-16 18:19:17 +08:00
//LogOut("发送消息,消息长度=> "+data.Length);
2021-12-10 23:38:40 +08:00
}
private void SendHeartbeat()
{
try
{
SendWithIndex(HeartbeatData);
}
catch (Exception ex)
{
//连接断开
OnCloseReady();
return;
}
2023-06-16 18:19:17 +08:00
//LogOut("发送心跳包");
2021-12-10 23:38:40 +08:00
}
object sendLock = new object();
2021-12-10 23:38:40 +08:00
/// <summary>
/// 发送数据并计数
/// </summary>
/// <param name="data"></param>
private void SendWithIndex(byte[] data)
{
lock (sendLock)
{
//增加发送计数
SendIndex = MaxSendIndexNum;
//发送数据
client.Send(data);
}
2021-12-10 23:38:40 +08:00
}
2024-01-08 16:19:09 +08:00
////拼接头长度
//private byte[] SendDataWithHead(byte[] message)
//{
2021-12-10 23:38:40 +08:00
2024-01-08 16:19:09 +08:00
// MemoryStream memoryStream = new MemoryStream();//创建一个内存流
2021-12-10 23:38:40 +08:00
2024-01-08 16:19:09 +08:00
// byte[] BagHead = BitConverter.GetBytes(message.Length + 4);//往字节数组中写入包头(包头自身的长度和消息体的长度)的长度
2021-12-10 23:38:40 +08:00
2024-01-08 16:19:09 +08:00
// memoryStream.Write(BagHead, 0, BagHead.Length);//将包头写入内存流
2021-12-10 23:38:40 +08:00
2024-01-08 16:19:09 +08:00
// memoryStream.Write(message, 0, message.Length);//将消息体写入内存流
2021-12-10 23:38:40 +08:00
2024-01-08 16:19:09 +08:00
// byte[] HeadAndBody = memoryStream.ToArray();//将内存流中的数据写入字节数组
2021-12-10 23:38:40 +08:00
2024-01-08 16:19:09 +08:00
// memoryStream.Close();//关闭内存
// memoryStream.Dispose();//释放资源
2021-12-10 23:38:40 +08:00
2024-01-08 16:19:09 +08:00
// return HeadAndBody;
//}
2021-12-10 23:38:40 +08:00
/// <summary>
/// 供外部调用 发送消息
/// </summary>
/// <param name="CMDID"></param>
/// <param name="data">序列化之后的数据</param>
public void SendToServer(int CMDID,byte[] data)
{
2023-06-16 18:19:17 +08:00
//LogOut("准备数据 CMDID=> "+CMDID);
/*
2021-12-10 23:38:40 +08:00
HunterNet_C2S _c2sdata = new HunterNet_C2S();
_c2sdata.HunterNetCoreCmdID = CMDID;
_c2sdata.HunterNetCoreData = ByteString.CopyFrom(data);
byte[] _finaldata = Serizlize(_c2sdata);
*/
byte[] _finaldata = HunterNet_C2S.CreatePkgData((ushort)CMDID, data);
2021-12-10 23:38:40 +08:00
SendToSocket(_finaldata);
}
2023-07-06 14:47:20 +08:00
#region
public delegate void OnReceiveDataHandler(int CMDID, int ERRCODE, byte[] data);
public delegate void OnConnectedHandler(bool IsConnected);
public delegate void OnCloseHandler();
public delegate void OnLogOutHandler(string Msg);
#endregion
2021-12-10 23:38:40 +08:00
2023-07-06 14:47:20 +08:00
public event OnConnectedHandler OnConnected;
public event OnReceiveDataHandler OnReceiveData;
public event OnCloseHandler OnClose;
2021-12-10 23:38:40 +08:00
/// <summary>
/// 网络库调试日志输出
/// </summary>
2023-07-06 14:47:20 +08:00
public event OnLogOutHandler OnLogOut;
2021-12-10 23:38:40 +08:00
/// <summary>
/// 做好处理的连接管理
/// </summary>
private void OnCloseReady()
{
2023-12-25 17:37:33 +08:00
if (bDetailedLog)
LogOut("关闭心跳包计数");
2021-12-10 23:38:40 +08:00
_heartTimer.Enabled = false;
2023-06-15 16:52:33 +08:00
_heartTimer.Elapsed -= CheckUpdatetimer_Elapsed;
2021-12-10 23:38:40 +08:00
LogOut("关闭连接");
//关闭Socket连接
client.Close();
2023-07-06 14:47:20 +08:00
OnClose?.Invoke();
2021-12-10 23:38:40 +08:00
}
/// <summary>
/// 主动关闭连接
/// </summary>
public void CloseConntect()
{
OnCloseReady();
}
private void DataCallBackReady(byte[] data)
{
//增加接收计数
RevIndex = MaxRevIndexNum;
//不处理心跳包
if (data.Length == 1 && data[0] == 0x00)
{
2023-06-16 18:19:17 +08:00
//LogOut("收到心跳包");
2021-12-10 23:38:40 +08:00
return;
}
/*
2021-12-10 23:38:40 +08:00
HunterNet_S2C _c2s = DeSerizlize<HunterNet_S2C>(data);
2023-07-06 14:47:20 +08:00
OnReceiveData(_c2s.HunterNetCoreCmdID, _c2s.HunterNetCoreERRORCode, _c2s.HunterNetCoreData.ToArray());
*/
HunterNet_S2C.AnalysisPkgData(data, out ushort CmdID, out ushort Error, out byte[] resultdata);
OnReceiveData(CmdID, Error, resultdata);
2021-12-10 23:38:40 +08:00
}
2023-12-14 18:15:04 +08:00
MemoryStream memoryStream = new MemoryStream();//开辟一个内存流
2021-12-10 23:38:40 +08:00
private void Recive(object o)
{
var client = o as Socket;
2023-12-14 18:15:04 +08:00
//MemoryStream memoryStream = new MemoryStream();//开辟一个内存流
2021-12-10 23:38:40 +08:00
while (true)
{
byte[] buffer = new byte[1024 * 1024 * 2];
int effective=0;
try
{
effective = client.Receive(buffer);
if (effective == 0)
{
continue;
}
}
catch(Exception ex)
{
//远程主机强迫关闭了一个现有的连接
OnCloseReady();
return;
//断开连接
}
memoryStream.Write(buffer, 0, effective);//将接受到的数据写入内存流中
byte[] getData = memoryStream.ToArray();//将内存流中的消息体写入字节数组
int StartIndex = 0;//设置一个读取数据的起始下标
while (true)
{
if (effective > 0)//如果接受到的消息不为0不为空
{
int HeadLength = 0;//包头长度(包头+包体)
if (getData.Length - StartIndex < 4)//包头接受不完整
{
HeadLength = -1;
}
else
{
//如果包头接受完整 转换成int类型的数值
HeadLength = BitConverter.ToInt32(getData, StartIndex);
}
//包头接受完整但是消息体不完整 //包头接受不完整
//↓↓↓↓↓↓↓↓ ↓↓↓
if (getData.Length - StartIndex < HeadLength || HeadLength == -1)
{
2023-12-14 18:15:04 +08:00
/*
2021-12-10 23:38:40 +08:00
memoryStream.Close();//关闭内存流
memoryStream.Dispose();//释放内存资源
memoryStream = new MemoryStream();//创建新的内存流
2023-12-14 18:15:04 +08:00
*/
//流复用的方式 不用重新new申请
memoryStream.Position = 0;
memoryStream.SetLength(0);
2021-12-10 23:38:40 +08:00
memoryStream.Write(getData, StartIndex, getData.Length - StartIndex);//从新将接受的消息写入内存流
break;
}
else
{
//把头去掉,就可以吃了,蛋白质是牛肉的六倍
2023-12-14 18:15:04 +08:00
//DataCallBackReady(getData.Skip(StartIndex+4).Take(HeadLength-4).ToArray());
int CoreLenght = HeadLength - 4;
2024-01-08 16:19:09 +08:00
//改为Array.Copy 提升效率
//byte[] retData = new byte[CoreLenght];
//Array.Copy(getData, StartIndex + 4, retData, 0, CoreLenght);
//DataCallBackReady(retData);
//用Span
Span<byte> getData_span = getData;
getData_span = getData_span.Slice(StartIndex + 4,CoreLenght);
DataCallBackReady(getData_span.ToArray());
2021-12-10 23:38:40 +08:00
StartIndex += HeadLength;//当读取一条完整的数据后,读取数据的起始下标应为当前接受到的消息体的长度(当前数据的尾部或下一条消息的首部)
}
}
}
}
}
private void CheckUpdatetimer_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
{
//接收服务器数据计数
RevIndex--;
if (RevIndex <= 0)
{
//判定掉线
OnCloseReady();
return;
}
//发送计数
SendIndex--;
if (SendIndex <= 0)//需要发送心跳包了
{
//重置倒计时计数
SendIndex = MaxSendIndexNum;
SendHeartbeat();
}
}
2021-12-10 23:38:40 +08:00
public void LogOut(string Msg)
{
//Console.WriteLine(Msg);
2023-07-06 14:47:20 +08:00
OnLogOut?.Invoke(Msg);
2021-12-10 23:38:40 +08:00
}
2023-06-02 18:24:54 +08:00
public Socket GetClientSocket()
{
return client;
}
2021-12-10 23:38:40 +08:00
}
}