📄 ceudp.cpp
字号:
// CEUDP.cpp: implementation of the CCEUDP class.
//
//////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include "UDPDemo.h"
#include "CEUDP.h"
#ifdef _DEBUG
#undef THIS_FILE
static char THIS_FILE[]=__FILE__;
#define new DEBUG_NEW
#endif
//////////////////////////////////////////////////////////////////////
// Construction/Destruction
//////////////////////////////////////////////////////////////////////
CCEUDP::CCEUDP()
{
}
CCEUDP::~CCEUDP()
{
}
/******************************************************************************
函数介绍:打开UDP通讯端口
入口参数:pWnd: 指定父窗体指针
localPort: 指定远程UDP端口
romoteHost:指定远程IP地址
remotePort:指定远程UDP端口
出口参数:(无)
返 回 值:1代表成功;-1,-2,-3等都代表失败
******************************************************************************/
DWORD CCEUDP::Open(CWnd* pWnd,int localPort,LPCTSTR remoteHost,int remotePort)
{
m_pOwnerWnd = pWnd;
// 加载winsock1.1动态链接库
WSADATA wsa;
if (WSAStartup(MAKEWORD(1,1),&wsa) != 0)
{
return -1; // 代表失败
}
// 创建UDP套接字
m_UDPSocket = socket(AF_INET,SOCK_DGRAM,IPPROTO_UDP);
if (m_UDPSocket == INVALID_SOCKET)
{
return -2;
}
SOCKADDR_IN localAddr;
localAddr.sin_family = AF_INET;
localAddr.sin_port = htons(localPort);
localAddr.sin_addr.s_addr = INADDR_ANY;
// 绑定地址
if(bind(m_UDPSocket,(sockaddr*)&localAddr,sizeof(localAddr))!=0)
{
return -3;
}
// 设置非堵塞通讯
DWORD ul= 1;
ioctlsocket(m_UDPSocket,FIONBIO,&ul);
// 创建一个线程退出事件
m_ExitThreadEvent = CreateEvent(NULL,TRUE,FALSE,NULL);
// 创建通讯线程
AfxBeginThread(RecvThread,this);
m_RemoteAddr.sin_family = AF_INET;
m_RemoteAddr.sin_port = htons(remotePort);
// 此处要将双字节转换成单字节
char ansiRemoteHost[255];
ZeroMemory(ansiRemoteHost, 255);
WideCharToMultiByte(CP_ACP,WC_COMPOSITECHECK,remoteHost,wcslen(remoteHost),
ansiRemoteHost,wcslen(remoteHost),NULL,NULL);
m_RemoteAddr.sin_addr.s_addr = inet_addr(ansiRemoteHost);
return 1;
}
/******************************************************************************
函数介绍:关闭UDP通讯端口
入口参数:(无)
出口参数:(无)
返 回 值:1代表成功;-1,-2等都代表失败
******************************************************************************/
DWORD CCEUDP::Close(void)
{
// 设置通讯线程退出事件,通知线程退出
SetEvent(m_ExitThreadEvent);
Sleep(2000);
// 关闭线程句柄
CloseHandle(m_ExitThreadEvent);
// 关闭socket
if (closesocket(m_UDPSocket) == SOCKET_ERROR)
{
return -1;
}
// 释放socket资源
if (WSACleanup() == SOCKET_ERROR)
{
return -2;
}
return 1;
}
/******************************************************************************
函数介绍:发送数据
入口参数:buf:缓冲区数据
len:缓冲数据长度
出口参数:(无)
返 回 值:发送成功代表实际发送的字节数,否则返回-1
******************************************************************************/
DWORD CCEUDP::SendData(const char *buf, int len)
{
int nBytes = 0;
int nSendBytes=0;
int nSumBytes =0;
int nErrorCode;
// 生成UDP数据包
UDPData sendData;
CopyMemory(sendData.FrameHead, FRAMEHEAD, 4);
sendData.DataPackageLen = len;
CopyMemory(sendData.FrameTail, FRAMETAIL, 4);
nSumBytes = len + 12;
char * sendBuf;
sendBuf = new char[nSumBytes];
CopyMemory(sendBuf, sendData.FrameHead, 4);
CopyMemory(sendBuf + 4, &(sendData.DataPackageLen), 4);
CopyMemory(sendBuf + 8, buf, len);
CopyMemory(sendBuf + 8 + len, sendData.FrameTail, 4);
// 发送数据
while (nSendBytes < nSumBytes)
{
nBytes = sendto(m_UDPSocket, sendBuf + nSendBytes, nSumBytes - nSendBytes, 0,
(sockaddr*)&m_RemoteAddr, sizeof(m_RemoteAddr));
if (nBytes == SOCKET_ERROR )
{
nErrorCode = WSAGetLastError();
m_OnUdpError(m_pOwnerWnd,nErrorCode);
return -1;
}
if (nSendBytes == nSumBytes)
{
break;
}
Sleep(1000);
nSendBytes = nSendBytes + nBytes;
}
delete[] sendBuf;
return nSendBytes;
}
/******************************************************************************
函数介绍:接收线程函数
入口参数:lparam : 指传进线程的参数
出口参数:(无)
返 回 值:无意义。
******************************************************************************/
UINT CCEUDP::RecvThread(LPVOID lparam)
{
CCEUDP *pSocket;
pSocket = (CCEUDP*)lparam;
fd_set fdRead;
int ret;
TIMEVAL aTime;
char * recvBuf=NULL;
aTime.tv_sec = 1;
aTime.tv_usec = 0;
SOCKADDR_IN tmpAddr;
int tmpRecvLen;
int recvLen;
int iErrorCode;
char * recvedBuf = NULL;
int recvedBufLen;
while (TRUE)
{
// 收到退出事件,结束线程
if (WaitForSingleObject(pSocket->m_ExitThreadEvent,0) == WAIT_OBJECT_0)
{
break;
}
// 将set初始化空集合
FD_ZERO(&fdRead);
// 将pSocket->m_UDPSocket套接字添加到集合中
FD_SET(pSocket->m_UDPSocket, &fdRead);
// 调用select函数,判断套接字I/O状态
ret = select(0, &fdRead, NULL, NULL, &aTime);
if (ret == SOCKET_ERROR)
{
iErrorCode = WSAGetLastError();
pSocket->m_OnUdpError(pSocket->m_pOwnerWnd, iErrorCode);
break;
}
if (ret > 0)
{
if (FD_ISSET(pSocket->m_UDPSocket, &fdRead))
{
tmpAddr.sin_family=AF_INET;
tmpAddr.sin_port = htons(pSocket->m_RemoteAddr.sin_port);
tmpAddr.sin_addr.s_addr =INADDR_ANY;
tmpRecvLen = sizeof(tmpAddr);
recvBuf = new char[512];
recvedBuf = new char[512];
ZeroMemory(recvBuf,512);
ZeroMemory(recvedBuf,512);
recvLen = recvfrom(pSocket->m_UDPSocket,recvBuf, 512,0,(SOCKADDR*)&tmpAddr,&tmpRecvLen);
if (recvLen == SOCKET_ERROR)
{
iErrorCode = WSAGetLastError();
pSocket->m_OnUdpError(pSocket->m_pOwnerWnd,iErrorCode);
break;
}
else if (recvLen == 0)
{
iErrorCode = WSAGetLastError();
pSocket->m_OnUdpError(pSocket->m_pOwnerWnd,iErrorCode);
break;
}
else
{
// 此处添加解析程序,将接收到的数据解析后,
if (pSocket->HandlePackage(recvBuf, recvLen, recvedBuf, recvedBufLen))
{
pSocket->m_OnUdpRecv(pSocket->m_pOwnerWnd, recvedBuf, recvedBufLen, (SOCKADDR*)&tmpAddr);
}
delete []recvBuf;
recvBuf = NULL;
delete []recvedBuf;
recvedBuf = NULL;
}
}
}
}
return 0;
}
/******************************************************************************
函数介绍:处理接收到的数据包
入口参数:inBuf: 待处理的缓冲区
inBufLen:待处理的缓冲区长度
出口参数
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -