2024-04-15 15:19:11 +08:00
|
|
|
|
//using HunterProtobufCore;
|
|
|
|
|
using System;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Net;
|
|
|
|
|
using System.Net.Sockets;
|
|
|
|
|
using System.Threading;
|
2024-10-18 16:55:40 +08:00
|
|
|
|
using static HaoYueNet.ClientNetwork.BaseData;
|
2024-04-15 15:19:11 +08:00
|
|
|
|
|
2024-10-18 16:55:40 +08:00
|
|
|
|
namespace HaoYueNet.ClientNetwork
|
2024-04-15 15:19:11 +08:00
|
|
|
|
{
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
|
|
//响应倒计时计数
|
2024-10-18 16:55:40 +08:00
|
|
|
|
private static int RevIndex=0;
|
2024-04-15 15:19:11 +08:00
|
|
|
|
//发送倒计时计数
|
2024-10-18 16:55:40 +08:00
|
|
|
|
private static int SendIndex=0;
|
2024-04-15 15:19:11 +08:00
|
|
|
|
|
|
|
|
|
//计时器间隔
|
|
|
|
|
private static int TimerInterval = 3000;
|
|
|
|
|
|
|
|
|
|
private System.Timers.Timer _heartTimer;
|
|
|
|
|
|
|
|
|
|
public static string LastConnectIP;
|
|
|
|
|
public static int LastConnectPort;
|
|
|
|
|
public bool bDetailedLog = false;
|
|
|
|
|
|
2024-10-18 16:55:40 +08:00
|
|
|
|
public bool Init(string IP, int port,bool isHadDetailedLog = true, bool bBindReuseAddress = false,int bBindport = 0,AddressFamily addressFamily = AddressFamily.InterNetwork)
|
2024-04-15 15:19:11 +08:00
|
|
|
|
{
|
|
|
|
|
LogOut("==>初始化网络核心");
|
|
|
|
|
|
|
|
|
|
bDetailedLog = isHadDetailedLog;
|
|
|
|
|
RevIndex = MaxRevIndexNum;
|
|
|
|
|
SendIndex = MaxSendIndexNum;
|
|
|
|
|
|
2024-10-18 16:55:40 +08:00
|
|
|
|
client = new Socket(addressFamily, SocketType.Stream, ProtocolType.Tcp);
|
2024-04-15 15:19:11 +08:00
|
|
|
|
if (bBindReuseAddress)
|
|
|
|
|
{
|
|
|
|
|
client.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.ReuseAddress, true);
|
|
|
|
|
IPEndPoint ipe = new IPEndPoint(IPAddress.Any, Convert.ToInt32(bBindport));
|
|
|
|
|
client.Bind(ipe);
|
|
|
|
|
}
|
|
|
|
|
LastConnectIP = IP;
|
|
|
|
|
LastConnectPort = port;
|
|
|
|
|
return Connect(IP, port);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Connect(string IP, int port)
|
|
|
|
|
{
|
|
|
|
|
//带回调的
|
|
|
|
|
try
|
|
|
|
|
{
|
2024-10-18 16:55:40 +08:00
|
|
|
|
if(bDetailedLog)
|
2024-04-15 15:19:11 +08:00
|
|
|
|
LogOut("连接到远程IP " + IP + ":" + port);
|
|
|
|
|
else
|
|
|
|
|
LogOut("连接到远程服务");
|
|
|
|
|
|
|
|
|
|
client.Connect(IP, port);
|
|
|
|
|
Thread thread = new Thread(Recive);
|
|
|
|
|
thread.IsBackground = true;
|
|
|
|
|
thread.Start(client);
|
|
|
|
|
int localport = ((IPEndPoint)client.LocalEndPoint).Port;
|
|
|
|
|
|
|
|
|
|
if (bDetailedLog)
|
|
|
|
|
LogOut($"连接成功!连接到远程IP->{IP}:{port} | 本地端口->{localport}");
|
|
|
|
|
else
|
|
|
|
|
LogOut("连接成功!");
|
|
|
|
|
|
|
|
|
|
if (_heartTimer == null)
|
|
|
|
|
{
|
|
|
|
|
_heartTimer = new System.Timers.Timer();
|
|
|
|
|
}
|
|
|
|
|
_heartTimer.Interval = TimerInterval;
|
|
|
|
|
_heartTimer.Elapsed += CheckUpdatetimer_Elapsed;
|
|
|
|
|
_heartTimer.AutoReset = true;
|
|
|
|
|
_heartTimer.Enabled = true;
|
|
|
|
|
|
|
|
|
|
if (bDetailedLog)
|
|
|
|
|
LogOut("开启心跳包检测");
|
|
|
|
|
|
|
|
|
|
OnConnected?.Invoke(true);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
if (bDetailedLog)
|
|
|
|
|
LogOut("连接失败:" + ex.ToString());
|
|
|
|
|
else
|
|
|
|
|
LogOut("连接失败");
|
|
|
|
|
|
|
|
|
|
OnConnected?.Invoke(false);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
~NetworkHelperCore()
|
|
|
|
|
{
|
|
|
|
|
client.Close();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void SendToSocket(byte[] data)
|
|
|
|
|
{
|
|
|
|
|
//已拼接包长度,这里不再需要拼接长度
|
|
|
|
|
//data = SendDataWithHead(data);
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
SendWithIndex(data);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
//连接断开
|
|
|
|
|
OnCloseReady();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
//LogOut("发送消息,消息长度=> "+data.Length);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void SendHeartbeat()
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
SendWithIndex(HeartbeatData);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
//连接断开
|
|
|
|
|
OnCloseReady();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
//LogOut("发送心跳包");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
object sendLock = new object();
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 发送数据并计数
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="data"></param>
|
|
|
|
|
private void SendWithIndex(byte[] data)
|
|
|
|
|
{
|
2024-10-18 16:55:40 +08:00
|
|
|
|
lock (sendLock)
|
2024-04-15 15:19:11 +08:00
|
|
|
|
{
|
|
|
|
|
//增加发送计数
|
|
|
|
|
SendIndex = MaxSendIndexNum;
|
|
|
|
|
//发送数据
|
|
|
|
|
client.Send(data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 供外部调用 发送消息
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="CMDID"></param>
|
|
|
|
|
/// <param name="data">序列化之后的数据</param>
|
2024-10-18 16:55:40 +08:00
|
|
|
|
public void SendToServer(int CMDID,byte[] data)
|
2024-04-15 15:19:11 +08:00
|
|
|
|
{
|
|
|
|
|
byte[] _finaldata = HunterNet_C2S.CreatePkgData((ushort)CMDID, data);
|
|
|
|
|
SendToSocket(_finaldata);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#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
|
|
|
|
|
|
|
|
|
|
public event OnConnectedHandler OnConnected;
|
|
|
|
|
public event OnReceiveDataHandler OnReceiveData;
|
|
|
|
|
public event OnCloseHandler OnClose;
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 网络库调试日志输出
|
|
|
|
|
/// </summary>
|
|
|
|
|
public event OnLogOutHandler OnLogOut;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 做好处理的连接管理
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void OnCloseReady()
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
if (bDetailedLog)
|
|
|
|
|
LogOut("关闭心跳包计数");
|
|
|
|
|
_heartTimer.Enabled = false;
|
|
|
|
|
_heartTimer.Elapsed -= CheckUpdatetimer_Elapsed;
|
|
|
|
|
LogOut("关闭连接");
|
|
|
|
|
//关闭Socket连接
|
|
|
|
|
client.Close();
|
|
|
|
|
OnClose?.Invoke();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 主动关闭连接
|
|
|
|
|
/// </summary>
|
|
|
|
|
public void CloseConntect()
|
|
|
|
|
{
|
|
|
|
|
OnCloseReady();
|
|
|
|
|
}
|
2024-10-18 16:55:40 +08:00
|
|
|
|
|
2024-04-15 15:19:11 +08:00
|
|
|
|
private void DataCallBackReady(byte[] data)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
//增加接收计数
|
|
|
|
|
RevIndex = MaxRevIndexNum;
|
|
|
|
|
|
|
|
|
|
//不处理心跳包
|
|
|
|
|
if (data.Length == 1 && data[0] == 0x00)
|
|
|
|
|
{
|
|
|
|
|
//LogOut("收到心跳包");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HunterNet_S2C.AnalysisPkgData(data, out ushort CmdID, out ushort Error, out byte[] resultdata);
|
|
|
|
|
OnReceiveData(CmdID, Error, resultdata);
|
|
|
|
|
}
|
|
|
|
|
|
2024-06-27 15:49:56 +08:00
|
|
|
|
MemoryStream reciveMemoryStream = new MemoryStream();//开辟一个反复使用的内存流
|
|
|
|
|
byte[] reciveBuffer = new byte[1024 * 1024 * 2];//开辟一个反复使用的byte[]
|
2024-04-15 15:19:11 +08:00
|
|
|
|
private void Recive(object o)
|
|
|
|
|
{
|
|
|
|
|
var client = o as Socket;
|
|
|
|
|
while (true)
|
|
|
|
|
{
|
2024-06-19 16:05:29 +08:00
|
|
|
|
int effective = 0;
|
2024-04-15 15:19:11 +08:00
|
|
|
|
try
|
|
|
|
|
{
|
2024-06-19 16:05:29 +08:00
|
|
|
|
effective = client.Receive(reciveBuffer);
|
|
|
|
|
if (effective == 0)//为0表示已经断开连接
|
2024-04-15 15:19:11 +08:00
|
|
|
|
{
|
2024-06-19 16:05:29 +08:00
|
|
|
|
//清理数据
|
|
|
|
|
reciveMemoryStream.SetLength(0);
|
|
|
|
|
reciveMemoryStream.Seek(0, SeekOrigin.Begin);
|
|
|
|
|
//远程主机强迫关闭了一个现有的连接
|
|
|
|
|
OnCloseReady();
|
|
|
|
|
return;
|
2024-04-15 15:19:11 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2024-10-18 16:55:40 +08:00
|
|
|
|
catch(Exception ex)
|
2024-04-15 15:19:11 +08:00
|
|
|
|
{
|
2024-06-19 16:05:29 +08:00
|
|
|
|
//清理数据
|
|
|
|
|
reciveMemoryStream.SetLength(0);
|
|
|
|
|
reciveMemoryStream.Seek(0, SeekOrigin.Begin);
|
|
|
|
|
|
2024-04-15 15:19:11 +08:00
|
|
|
|
//远程主机强迫关闭了一个现有的连接
|
|
|
|
|
OnCloseReady();
|
|
|
|
|
return;
|
|
|
|
|
//断开连接
|
|
|
|
|
}
|
|
|
|
|
|
2024-06-19 16:05:29 +08:00
|
|
|
|
reciveMemoryStream.Write(reciveBuffer, 0, effective);//将接受到的数据写入内存流中
|
|
|
|
|
byte[] getData = reciveMemoryStream.ToArray();//将内存流中的消息体写入字节数组
|
2024-04-15 15:19:11 +08:00
|
|
|
|
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)
|
|
|
|
|
{
|
|
|
|
|
//流复用的方式 不用重新new申请
|
2024-06-19 16:05:29 +08:00
|
|
|
|
reciveMemoryStream.Position = 0;
|
|
|
|
|
reciveMemoryStream.SetLength(0);
|
2024-04-15 15:19:11 +08:00
|
|
|
|
|
2024-06-19 16:05:29 +08:00
|
|
|
|
reciveMemoryStream.Write(getData, StartIndex, getData.Length - StartIndex);//从新将接受的消息写入内存流
|
2024-04-15 15:19:11 +08:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
int CoreLenght = HeadLength - 4;
|
|
|
|
|
//用Span
|
2024-06-19 16:05:29 +08:00
|
|
|
|
Span<byte> getData_span = getData;
|
|
|
|
|
getData_span = getData_span.Slice(StartIndex + 4, CoreLenght);
|
|
|
|
|
DataCallBackReady(getData_span.ToArray());
|
2024-04-15 15:19:11 +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();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void LogOut(string Msg)
|
|
|
|
|
{
|
|
|
|
|
OnLogOut?.Invoke(Msg);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Socket GetClientSocket()
|
|
|
|
|
{
|
|
|
|
|
return client;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|