当前位置: 首页 > news >正文

《网络编程基础之完成端口模型》

【完成端口模型导读】完成端口模型,算得上是真正的异步网络IO模型吧,相对于其它网络IO模型,操作系统通知我们的时候,要么就是连接已经帮我建立好,客户端套接字帮我们准备好;要么就是数据已经接收完成;要么就是本端的数据已经发送出去了。我们只需准备接收数据的容器以及客户端的套接字即可。

     

1、重难点分析

     使用完成端口模型的几个主要步骤:

1、创建完成端口模型2、创建监听socket,并将监听socket绑定到指定的IP地址和端口3、将监听socket绑定到完成端口模型上去

    以上步骤可以使用如下的示例代码进行概述:

HANDLE hCompletePort = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, 0);//监听socket,必须使用WSASocket接口创建,而且要加上WSA_FLAG_OVERLAPPED标记                             HANDLE listenSocket = WSASocket(AF_INET, SOCK_STREAM, 0, NULL, 0, WSA_FLAG_OVERLAPPED);// 将监听Socket绑定到完成端口struct per_socket_context listenContext;//per_socket_context,下文会介绍。if( NULL== CreateIoCompletionPort((HANDLE)listenSocket,         hCompletePort,(ULONG_PTR)&listenContext,0))  {  return;}  //开启监听struct sockaddr_in serverAddress;// 填充地址信息ZeroMemory((char *)&serverAddress, sizeof(serverAddress));serverAddress.sin_family = AF_INET;// 这里可以绑定任何可用的IP地址,或者绑定一个指定的IP地址 serverAddress.sin_addr.s_addr = htonl(INADDR_ANY);                            serverAddress.sin_port = htons(8000);if (SOCKET_ERROR = bind(listenSocket, (struct sockaddr_in*)&serverAddress,sizeof(serverAddress))){return;}if (listen(listenSocket, SOMAXCONN) == SOCKET_ERROR){return;}                     
          正如前文所说,完成端口模型帮我把连接建立,我们只需要准备好客户端套接字即可,可以这么理解吧!应用层只需要投递个接收连接的请求,等到操作系统通知我们的时候,连接已经建立好,客户端套接字已经生效,它表示一个合法的"客户端对象",此时可以通过这个客户端套接字和对端建立连接,进行数据通信。

     那投递接收连接请求的接口是哪个?  就是AcceptEx。   

1、接口声明
BOOL AcceptEx(_In_  SOCKET sListenSocket, //监听套接字_In_  SOCKET sAcceptSocket, //客户端套接字,使用完,下次投递请求的时候需再补充一个_In_  PVOID  lpOutputBuffer, //接收在新连接上发送的第一个数据块_In_  DWORD  dwReceiveDataLength, //实际接收数据的字节数_In_  DWORD  dwLocalAddressLength, //本地地址信息保留的字节数_In_  DWORD  dwRemoteAddressLength, //远程地址信息保留的字节数_Out_ LPDWORD lpdwBytesReceived, //指向接收接收字节计数的DWORD指针_In_  LPOVERLAPPED lpOverlapped //用于处理请求的 OVERLAPPED 结构。必须指定此参数;它不能为 NULL
);2、接口返回值AcceptEx函数成功完成,返回值为TRUE 。函数失败,AcceptEx返回FALSE。可调用WSAGetLastError返回扩展错误信息。如果 WSAGetLastError 返回 ERROR_IO_PENDING,则表示操作已成功启动,仍在进行中。 如果是其它错误,说明真的出错了。 

  

     虽然微软提供了接口的声明,根据msdn的官方声明,我们还是使用其它的方式去获取AcceptEx函数指针,具体的方式如下:


DWORD dwBytes = 0;  
if(SOCKET_ERROR == WSAIoctl(listenSocket, //任意一个有效的socket即可 SIO_GET_EXTENSION_FUNCTION_POINTER, &GuidAcceptEx, sizeof(GuidAcceptEx), &m_lpfnAcceptEx, sizeof(m_lpfnAcceptEx), &dwBytes, NULL, NULL))  
{return;  
}

      m_lpfnAcceptEx就是我们拿到的AcceptEx函数指针,我们可以使用它来投递一个接收连接的请求。

       好!介绍完投递接收连接请求的接口,我们再看看投递接收数据的接口又是怎样的?


int WSAAPI WSARecv(SOCKET s, // 客户套接字LPWSABUF lpBuffers,//指向 WSABUF 结构数组的指针DWORD  dwBufferCount, //lpBuffers数组中的WSABUF结构数。LPDWORD lpNumberOfBytesRecvd,//接收的数据数(以字节为单位)的指针LPDWORD  lpFlags,//默认为0LPWSAOVERLAPPED  lpOverlapped, //overlapped结构指针LPWSAOVERLAPPED_COMPLETION_ROUTINE lpCompletionRoutine
);

 

   那如何投递接收数据的请求:

// 初始化变量
DWORD dwFlags = 0;
DWORD dwBytes = 0;
WSABUF *Wbuf;
OVERLAPPED *pol = &pIoContext->m_Overlapped;
pIoContext->ResetBuffer();
pIoContext->m_OpType = RECV_POSTED;
int nBytesRecv = WSARecv(clientSock, Wbuf, 1, &dwBytes, &dwFlags, pol, NULL );if ((SOCKET_ERROR == nBytesRecv) && (WSA_IO_PENDING != WSAGetLastError()))return;

   投递发送数据的请求也是类似:

// 初始化变量
DWORD dwFlags = 0;
DWORD dwBytes = 0;
WSABUF *p_wbuf;
OVERLAPPED *pol = &pIoContext->m_Overlapped;
pIoContext->ResetBuffer();
pIoContext->m_OpType = SEND_POSTED;
// 初始化完成后,投递WSARecv请求
int nBytesSend = WSASend(ClientSock, p_wbuf, 1, &dwBytes, &dwFlags, p_ol, NULL);
// 如果返回值错误,并且错误的代码并非是Pending的话,那就说明这个重叠请求失败了
if ((SOCKET_ERROR == nBytesSend) && (WSA_IO_PENDING != WSAGetLastError()))
{return ;
}

     投递成功后,如果对端有数据发送过来,那么完成端口的工作线程便会被唤醒,我们将从工作线程拿到对端发送过来的数据;亦或数据成功发送给对端,那么工作线程也会被唤醒,这个时候是继续投递发送请求还是投递一个接收数据的请求?就看自己的业务需要了。

    正如上文所说,投递完请求,工作线程被唤醒,这个时候如何区分是"发送"成功还是"接收"成功?是监听套接字的事件还是客户端套接字的事件。介绍完如何区分这些差别之前,我们先看看工作线程是被什么API给挂起的?以至于有事件来时能被唤醒。

BOOL WINAPI GetQueuedCompletionStatus(_In_  HANDLE  CompletionPort, //完成端口句柄_Out_ LPDWORD lpNumberOfBytes, //指向接收已完成 I/O 操作中传输的字节数的变量的指针。/*指向和完成端口相关附加数据,和CreateIoCompletionPort第三个参数是同一个指针,指向同一份内存。因此在将socket绑定到完成端口时,可以在lpCompletionKey中指定socket的类型那么工作线程就知道当前的事件是连接套接字产生的还是客户端套接字产生的*/_Out_ PULONG_PTR lpCompletionKey, _Out_ LPOVERLAPPED *lpOverlapped,_In_  DWORD  dwMilliseconds
);

    像lpCompletionKey和socket相关的结构体指针,我们称之为单socket数据,具体的用法如下:

typedef struct _PER_SOCKET_CONTEXT
{//每一个客户端连接的SocketSOCKET  m_Socket;     //客户端的地址SOCKADDR_IN m_ClientAddr;//客户端网络操作的上下文数据,CArray<_PER_IO_CONTEXT*> m_arrayIoContext;             
};//对于侦听socket
_PER_SOCKET_CONTEXT listenSocketContext;
listenSocketContext.s = ListenSocket;
CreateIoCompletionPort(ListenSocket, m_hIOCompletionPort, (ULONG_PTR)&listenSocketContext,0);//对于普通客户端连接socket
_PER_SOCKET_CONTEXT clientSocketContext;
clientSocketContext.s = acceptSocket;
CreateIoCompletionPort(acceptSocket, m_hIOCompletionPort,(ULONG_PTR)&clientSocketContext,0);

     那么工作线程可以从对应的socketContext中解析出socket的值,具体是监听socket还是客户端socket,那可以做区分处理。

DWORD ThreadFunction()
{OVERLAPPED *pOverlapped = NULL;_PER_SOCKET_CONTEXT *pSocketContext = NULL;DWORD dwBytesTransfered = 0;BOOL bReturn = GetQueuedCompletionStatus(m_hIOCompletionPort,&dwBytesTransfered, (PULONG_PTR)&pSocketContext, &pOverlapped, INFINITE);if (((SOME_STRUCT*)pSocketContext)->s == 监听socket){//新连接接收成功,做一些操作}else  //客户端套接字{if (事件类型 == 收到了一份数据){//解析数据}else if (事件类型 == 数据发送成功了){//继续投递发送数据请求}}
}

    上述伪码中,假如是客户端套接字的事件,那么如何去区分事件类型,到底是"数据发送成功"还是"数据接收成功"?PER_SOCKET_DATA中是没有表明事件类型的。具体是发送数据还是接收数据,我们在投递发送请求或者接收请求的时候,是可以标注的,我们再回过头看看WSARecv接口的声明。


int WSAAPI WSARecv(SOCKET s, // 客户套接字......LPWSAOVERLAPPED  lpOverlapped, //overlapped结构指针LPWSAOVERLAPPED_COMPLETION_ROUTINE lpCompletionRoutine
);
 

    OVERLAPPED结构体指针lpOverlapped,基于指针的伸缩性,我们可以通过偏移访问指定范围的内存数据,假如我们在OVERLAPPED结构后面附加一些额外的数据用于标识当前的IO行为是"接收"还是"发送",那么工作线程被唤醒时,就可以通过lpOverlapped指针取出附加在OVERLAPPED后面的数据了,那么具体的事件类型就能很轻松获取到了。针对这种标识IO行为的结构体,我们称为PER_IO_DATA。


typedef struct _PER_IO_CONTEXT
{// 每一个重叠网络操作的重叠结构(针对每一个Socket的每一个操作,都要有一个)              OVERLAPPED m_Overlapped;// 这个网络操作所使用的SocketSOCKET m_sockAccept;  // WSA类型的缓冲区,用于给重叠操作传参数的WSABUF m_wsaBuf;  // 这个是WSABUF里具体存字符的缓冲区char  m_szBuffer[MAX_BUFFER_LEN]; // 标识网络操作的类型(对应上面的枚举), 发送或者接收事件OPERATION_TYPE m_OpType;                                    
};

     那上文提到的工作线程就可以这样做改造了:

DWORD ThreadFunction()
{OVERLAPPED *pOverlapped = NULL;_PER_SOCKET_CONTEXT *pSocketContext = NULL;DWORD dwBytesTransfered = 0;BOOL bReturn = GetQueuedCompletionStatus(m_hIOCompletionPort,&dwBytesTransfered, (PULONG_PTR)&pSocketContext, &pOverlapped, INFINITE);if (((SOME_STRUCT*)pSocketContext)->s == 监听socket){//新连接接收成功,做一些操作}else  //客户端套接字{//推荐使用微软封装的接口进行转换PER_IO_CONTEXT* pIoContext = CONTAINING_RECORD(pOverlapped, PER_IO_CONTEXT, m_Overlapped);if (pIOContext->Type == 接收){//解析数据}else if (pIOContext->Type == 发送){}}
}
有事件产生时,工作线程会被唤醒,假如我们想主动退出工作线程,那这个时候该如何唤醒挂在GetQueuedCompletionStatus接口的线程,既然有Get接口,那肯定就有Post接口;微软提供如下接口用于唤醒挂在GetQueuedCompletionStatus接口的线程。

BOOL WINAPI PostQueuedCompletionStatus(_In_  HANDLE CompletionPort,_In_  DWORD dwNumberOfBytesTransferred,_In_  ULONG_PTR dwCompletionKey,_In_opt_ LPOVERLAPPED lpOverlapped
);
 

    假设我们将第三个参数dwCompletionKey指向一个特殊的标识符(EXIT_CODE),那么工作线程被唤醒时,是不是可以从dwCompletionKey解析出这个特殊的标识符(EXIT_CODE),随后退出工作线程。

PostQueuedCompletionStatus(m_hIOCompletionPort, 0, (DWORD)EXIT_CODE, NULL);DWORD ThreadFunction()
{OVERLAPPED           *pOverlapped = NULL;PER_SOCKET_CONTEXT   *pSocketContext = NULL;DWORD                dwBytesTransfered = 0;BOOL bReturn = GetQueuedCompletionStatus(m_hIOCompletionPort, &dwBytesTransfered, (PULONG_PTR)&pSocketContext, &pOverlapped, INFINITE);//收到退出标志,退出线程if ( EXIT_CODE==(DWORD)pSocketContext ){return 0;}......
}      

2、完整示例代码

 IOCPModel.cpp

#include "CIOCPModel.h"#define DEFAULT_NUM_OF_PROCESSOR 2#define RELEASE_HANDLE(x) if(x != NULL && x != INVALID_HANDLE_VALUE ) if(x != NULL){ CloseHandle(x); x = NULL;}
#define RELEASE_SOCKET(x) if(x!= NULL && x!= INVALID_SOCKET){closesocket(x); x = NULL;}using namespace std;DWORD __stdcall CIOCPModel::_WorkThread(LPVOID lpParam)
{THREADPARAMS_WORKER *pThreadParam = (PTHREADPARAMS_WORKER)lpParam;CIOCPModel *pIoModel = pThreadParam->pIOCPModel;size_t ThreadNo = pThreadParam->dwThreadNo;//线程等到系统的IO通知,需要准备三个数据结构OVERLAPPED		   *pOverLapped = NULL;PER_SOCKET_CONTEXT *perSocket   = NULL;DWORD				dwByteTransfered = 0;while (WAIT_OBJECT_0 != WaitForSingleObject(pIoModel->m_hShutDownEvent, 0)){bool bFlag = GetQueuedCompletionStatus(pIoModel->m_hIOCompletionPort,&dwByteTransfered, (PULONG_PTR)&perSocket, &pOverLapped, INFINITE);OVERLAPPED m_Overlapped;//先处理主动退出的逻辑4if(EXIT_CODE == (DWORD)perSocket){break;}if (!bFlag){continue;}else{PER_IO_CONTEXT *perIoContext = CONTAINING_RECORD(pOverLapped, PER_IO_CONTEXT, m_Overlapped);if ((dwByteTransfered == 0) &&(perIoContext->m_OperatorType == RECV_POSTED|| perIoContext->m_OperatorType == SEND_POSTED)){cout << "Client is closed." << endl;pIoModel->_RemoveContextList(perSocket);continue;}switch (perIoContext->m_OperatorType){case RECV_POSTED://pIoModel->_PostRecv(perIoContext);pIoModel->_DoRecv(perSocket, perIoContext);break;case SEND_POSTED:pIoModel->_PostSend(perIoContext);break;case ACCEPT_POSTED:pIoModel->_DoAccept(perSocket, perIoContext);break;default:break;}}}if (pThreadParam != NULL){delete pThreadParam;}return 0;
}CIOCPModel::CIOCPModel():m_nThreads(0),m_hShutDownEvent(NULL),m_hIOCompletionPort(NULL),m_lpfnAcceptEx(NULL),m_lpfnGetAcceptExSockAddrs(NULL),m_WorkThread(NULL),m_strIP(DEFAULT_IP),m_port(DEFAULT_PORT),pListenSocketContext(NULL){}CIOCPModel::~CIOCPModel()
{this->stop();
}bool CIOCPModel::start()
{InitializeCriticalSection(&m_cs);m_hShutDownEvent = CreateEvent(NULL, TRUE, FALSE, NULL);if (this->_InitIOCPModel()){std::cout << "initialize IOCP Model success." << std::endl;}else{std::cout << "initialize IOCP Model failed." << std::endl;return false;}//初始化监听socketif (this->_InitializeListenSocket()){std::cout << "initialize Listen socket success" << std::endl;}else{std::cout << "initialize Listen socket error" << std::endl;return false;}return true;
}
void CIOCPModel::stop()
{//主动退出,OVERLAPPED			*pOverlapped = NULL;PER_SOCKET_CONTEXT  *perSocketContext = NULL;DWORD				dwByteTransfered = 0;if (pListenSocketContext != NULL && pListenSocketContext->m_Socket != INVALID_SOCKET){//先将m_hShutDownEvent设置为有信号的状态SetEvent(m_hShutDownEvent);for (int i = 0; i < m_nThreads; i++){PostQueuedCompletionStatus(m_hIOCompletionPort, dwByteTransfered, (DWORD)EXIT_CODE,pOverlapped);}WaitForMultipleObjects(m_nThreads, m_WorkThread, true, INFINITE);}
}
bool CIOCPModel::LoadSocketLib()
{return true;
}
void CIOCPModel::unLoadSocketLib()
{}
std::string CIOCPModel::GetLocalIP()
{char hostName[256] = { 0 };gethostname(hostName, 256);struct hostent FAR* lpHostEnt = gethostbyname(hostName);if (lpHostEnt == NULL)return "127.0.0.1";LPSTR lpAddr = lpHostEnt->h_addr_list[0];struct in_addr inAddr;memmove(&inAddr, lpAddr, 4);m_strIP  = inet_ntoa(inAddr);return m_strIP;
}
void CIOCPModel::setPort(const int &m_port)
{}bool CIOCPModel::_InitIOCPModel()
{//创建完成端口模型m_hIOCompletionPort = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, NULL, 0);if (m_hIOCompletionPort == NULL){std::cout << "创建完成端口模型失败" << std::endl;return false;}//初始化一个线程池,经验值:cpu的数量乘以2m_nThreads = 2 * _GetNoOfProcessor();m_WorkThread = new HANDLE[m_nThreads];DWORD nThreadId;for (int i = 0; i < m_nThreads; i++){THREADPARAMS_WORKER *pWorkThreadParam = new THREADPARAMS_WORKER;pWorkThreadParam->pIOCPModel = this;pWorkThreadParam->dwThreadNo = i + 1;m_WorkThread[i] = ::CreateThread(0, 0, _WorkThread, (LPVOID)pWorkThreadParam, 0, &nThreadId);}std::cout << "创建了 " << m_nThreads << " 个线程" << std::endl;return true;
}bool CIOCPModel::_InitializeListenSocket()
{GUID GuidAcceptEx			  = WSAID_ACCEPTEX;GUID GuidGetAcceptExSockAddrs = WSAID_GETACCEPTEXSOCKADDRS;struct sockaddr_in serverAddr;pListenSocketContext = new PER_SOCKET_CONTEXT;pListenSocketContext->m_Socket = WSASocket(AF_INET, SOCK_STREAM, 0, NULL,0, WSA_FLAG_OVERLAPPED);if (pListenSocketContext->m_Socket == INVALID_SOCKET){std::cout << "create listen socket error" << std::endl;return false;}//将监听套接字绑定到完成端口上if (NULL == CreateIoCompletionPort((HANDLE)pListenSocketContext->m_Socket,m_hIOCompletionPort, (DWORD)pListenSocketContext, 0)){std::cout << "Associate completion port to listen socket error" << std::endl;return false;}//绑定地址ZeroMemory(&serverAddr, sizeof(serverAddr));serverAddr.sin_addr.S_un.S_addr = htonl(INADDR_ANY);serverAddr.sin_port = htons(DEFAULT_PORT);serverAddr.sin_family = AF_INET;if (bind(pListenSocketContext->m_Socket, (struct sockaddr *)&serverAddr, sizeof(serverAddr)) == -1){std::cout << "bind error" << std::endl;return false;}if (-1 == listen(pListenSocketContext->m_Socket, SOMAXCONN)){std::cout << "listen error" << std::endl;return false;}DWORD dwByets = 0;if (WSAIoctl(pListenSocketContext->m_Socket,SIO_GET_EXTENSION_FUNCTION_POINTER,&GuidAcceptEx,sizeof(GuidAcceptEx),&m_lpfnAcceptEx,sizeof(m_lpfnAcceptEx),&dwByets,0,0) == -1){std::cout << "get acceptex error" << std::endl;return false;}if (WSAIoctl(pListenSocketContext->m_Socket,SIO_GET_EXTENSION_FUNCTION_POINTER,&GuidGetAcceptExSockAddrs,sizeof(GuidGetAcceptExSockAddrs),&m_lpfnGetAcceptExSockAddrs,sizeof(m_lpfnGetAcceptExSockAddrs),&dwByets,0,0) == -1){std::cout << "get GuidGetAcceptExSockAddrs error" << std::endl;return false;}//事先准备五个客户端套接字for (int i = 0; i < 10; i++){PER_IO_CONTEXT *PerIoContext = pListenSocketContext->GetNewIOContext();if (this->_PostAccept(PerIoContext) == false){std::cout << "post accept error" << std::endl;return false;}}return true;
}void CIOCPModel::_DeInitializeIOCPModel()
{DeleteCriticalSection(&m_cs);RELEASE_HANDLE(m_hShutDownEvent);for (int i = 0; i < m_nThreads; i++){RELEASE_HANDLE(this->m_WorkThread[i]);}delete []this->m_WorkThread;RELEASE_HANDLE(m_hIOCompletionPort);delete pListenSocketContext;return;
}bool CIOCPModel::_PostAccept(PER_IO_CONTEXT *pAcceptIoContext)
{//投递Accept请求,需要事先准备好客户端套接字assert(pListenSocketContext->m_Socket != INVALID_SOCKET);DWORD dwBytes = 0;pAcceptIoContext->m_OperatorType = ACCEPT_POSTED;WSABUF *wsaBuf = &pAcceptIoContext->m_wsaBuf;OVERLAPPED *p_ol = &pAcceptIoContext->m_Overlapped;pAcceptIoContext->m_AcceptSocket = WSASocket(AF_INET, SOCK_STREAM, 0, NULL,0, WSA_FLAG_OVERLAPPED);if (pAcceptIoContext->m_AcceptSocket == INVALID_SOCKET){std::cout << "create accept socket error" << std::endl;return false;}if (false == m_lpfnAcceptEx(pListenSocketContext->m_Socket, pAcceptIoContext->m_AcceptSocket,wsaBuf->buf, wsaBuf->len - (sizeof(SOCKADDR_IN) + 16) *2 ,sizeof(SOCKADDR_IN) + 16, sizeof(SOCKADDR_IN) + 16,&dwBytes, p_ol)){DWORD dwError = WSAGetLastError();if (dwError != WSA_IO_PENDING){std::cout << "Post Accept error" << std::endl;return false;}}return true;
}
bool CIOCPModel::_PostRecv(PER_IO_CONTEXT *PerContext)
{//投递Recv请求DWORD dwBytes = 0;DWORD dwFlag = 0;WSABUF *m_wsaBuf = &PerContext->m_wsaBuf;OVERLAPPED *o_lp = &PerContext->m_Overlapped;PerContext->ResetBuffer();PerContext->m_OperatorType = RECV_POSTED;int dwBytesReturn = WSARecv(PerContext->m_AcceptSocket, m_wsaBuf, 1, &dwBytes, &dwFlag, o_lp, NULL);if (dwBytesReturn == -1){if (WSAGetLastError() != WSA_IO_PENDING){std::cout << "post recv error " << WSAGetLastError() << std::endl;return false;}}//cout << "ThreadId: "<< GetCurrentThreadId() << "  recv from client: " << m_wsaBuf->buf << " Len: " << m_wsaBuf->len << endl;return true;
}bool CIOCPModel::_PostSend(PER_IO_CONTEXT *PerContext)
{DWORD dwBytes = 0;DWORD dwFlag = 0;WSABUF *m_wsaBuf = &PerContext->m_wsaBuf;OVERLAPPED *o_l = &PerContext->m_Overlapped;PerContext->m_OperatorType = SEND_POSTED;int dwBytesReturn = WSASend(PerContext->m_AcceptSocket, m_wsaBuf, 1, &dwBytes, dwFlag, o_l, NULL);if (dwBytesReturn == -1){if (WSAGetLastError() == WSA_IO_PENDING){cout << "Post send error" << endl;return false;}}return true;
}bool CIOCPModel::_DoAccept(PER_SOCKET_CONTEXT *PerSocket, PER_IO_CONTEXT *PerContext)
{SOCKADDR_IN *ClientAddr = NULL;SOCKADDR_IN *ServerAddr = NULL;int dwRemoteAddrLen = sizeof(SOCKADDR_IN);int dwLocalAddrLen = sizeof(SOCKADDR_IN);//把客户端和服务端的IP地址拿到,顺带把服务端发过来的第一组数据给拿到this->m_lpfnGetAcceptExSockAddrs(PerContext->m_wsaBuf.buf,PerContext->m_wsaBuf.len - (sizeof(SOCKADDR_IN) + 16) * 2,sizeof(SOCKADDR_IN) + 16, sizeof(SOCKADDR_IN) + 16,(LPSOCKADDR *)&ServerAddr, &dwLocalAddrLen, (LPSOCKADDR *)&ClientAddr, &dwRemoteAddrLen);std::cout << "客户端地址:" << inet_ntoa(ClientAddr->sin_addr) << " 端口: " << ClientAddr->sin_port << std::endl;std::cout << "服务端地址: " << inet_ntoa(ServerAddr->sin_addr) << "  端口: " << ServerAddr->sin_port << std::endl;std::cout << "客户端的发来的数据: " << PerContext->m_wsaBuf.buf << endl;PER_SOCKET_CONTEXT *PerSocketContext = new PER_SOCKET_CONTEXT;PerSocketContext->m_Socket = PerContext->m_AcceptSocket;memcpy(&PerSocketContext->m_ClientAddr, ClientAddr, sizeof(SOCKADDR_IN));//将Accept Socket和完成端口绑定在一起if (false == this->_AssociateWithIOCP(PerSocketContext)){std::cout << "Associate accept scket error" << std::endl;return false;}PER_IO_CONTEXT *PerIoContext				= PerSocketContext->GetNewIOContext();PerIoContext->ResetBuffer();PerIoContext->m_OperatorType = RECV_POSTED;PerIoContext->m_AcceptSocket = PerSocketContext->m_Socket;//开始投递一个Recv请求if (false == this->_PostRecv(PerIoContext)){if (WSAGetLastError() != WSA_IO_PENDING){cout << "Do Accept Post Recv failed." << WSAGetLastError() << endl;PerSocket->RemoveIOContext(PerIoContext);return false;}}//投递成功,将该请求添加到PerSocketContext队列this->_AddToContextList(PerSocket);//清空缓冲区PerIoContext->ResetBuffer();return this->_PostRecv(PerIoContext);
}bool CIOCPModel::_DoRecv(PER_SOCKET_CONTEXT *PerSocketContext, PER_IO_CONTEXT *PerIoContext)
{SOCKADDR_IN *ClientAddr = &PerSocketContext->m_ClientAddr;cout << "Client Addr: " << inet_ntoa(ClientAddr->sin_addr) << " Port: "<< ClientAddr->sin_port << "  "<< PerIoContext->m_wsaBuf.buf << " len: "<< PerIoContext->m_wsaBuf.len << endl;//开始投递下一个请求return this->_PostRecv(PerIoContext);
}void CIOCPModel::_AddToContextList(PER_SOCKET_CONTEXT *pSocketContext)
{EnterCriticalSection(&m_cs);m_ClientContextDeque.push_back(pSocketContext);LeaveCriticalSection(&m_cs);return;
}void CIOCPModel::_RemoveContextList(PER_SOCKET_CONTEXT *pSocketContext)
{EnterCriticalSection(&m_cs);PER_SOCKET_CONTEXT *pTempSocket = m_ClientContextDeque.front();m_ClientContextDeque.pop_front();delete pTempSocket;LeaveCriticalSection(&m_cs);
}void CIOCPModel::_ClearContextList()
{EnterCriticalSection(&m_cs);m_ClientContextDeque.clear();LeaveCriticalSection(&m_cs);
}bool CIOCPModel::_AssociateWithIOCP(PER_SOCKET_CONTEXT *pSocketContext)
{//将对应的句柄绑定到完成端口中区HANDLE hCompletePort = CreateIoCompletionPort((HANDLE)pSocketContext->m_Socket,m_hIOCompletionPort, (DWORD)pSocketContext, 0);if (hCompletePort == NULL){cout << "Associate Io Complettion Port failed. " << endl;return false;}return true;
}bool CIOCPModel::HandleError(PER_SOCKET_CONTEXT *pSocketContext, const DWORD &dwError)
{return true;
}int CIOCPModel::_GetNoOfProcessor()
{SYSTEM_INFO si;GetSystemInfo(&si);return si.dwNumberOfProcessors;
}bool CIOCPModel::_IsSocketAlive(SOCKET s)
{int nBytesSend = send(s, "a", 1, 0);if (nBytesSend == -1)return false;return true;
}

IOCPModel.h
 

#pragma once
#ifndef _C_IOCP_MODEL_H_
#define _C_IOCP_MODEL_H_
#include <iostream>
#include <WinSock2.h>
#include <algorithm>
#include <deque>
#include <vector>
#include <assert.h>
#include <MSWSock.h>
#pragma comment(lib,"ws2_32.lib")#define MAX_BUFFER_LEN	8192
#define DEFAULT_PORT	12345
#define DEFAULT_IP		"127.0.0.1"#define EXIT_CODE 0x111typedef enum _OPERATION_TYPE
{ACCEPT_POSTED,SEND_POSTED,RECV_POSTED,NULL_POSTED
}OPERATION_TYPE;typedef struct _PER_IO_CONTEXT
{OVERLAPPED m_Overlapped;SOCKET	   m_AcceptSocket;WSABUF	   m_wsaBuf;char	   m_szBuf[MAX_BUFFER_LEN];OPERATION_TYPE		m_OperatorType;_PER_IO_CONTEXT(){ZeroMemory(&m_Overlapped, sizeof(m_Overlapped));ZeroMemory(m_szBuf, MAX_BUFFER_LEN);m_AcceptSocket	= INVALID_SOCKET;m_wsaBuf.buf	= m_szBuf;m_wsaBuf.len	= MAX_BUFFER_LEN;m_OperatorType	= NULL_POSTED;}~_PER_IO_CONTEXT(){if (m_AcceptSocket != INVALID_SOCKET){closesocket(m_AcceptSocket);m_AcceptSocket = INVALID_SOCKET;}}void ResetBuffer(){ZeroMemory(m_szBuf, MAX_BUFFER_LEN);}}PER_IO_CONTEXT,*PPER_IO_CONTEXT;typedef struct _PER_SOCKET_CONTEXT
{SOCKET m_Socket;SOCKADDR_IN m_ClientAddr;std::deque<PER_IO_CONTEXT *> m_PerIoContextArr;_PER_SOCKET_CONTEXT(){m_Socket = INVALID_SOCKET;memset(&m_ClientAddr, 0, sizeof(m_ClientAddr));}~_PER_SOCKET_CONTEXT(){if (m_Socket != INVALID_SOCKET){closesocket(m_Socket);m_Socket = INVALID_SOCKET;}for (int i = 0; i < m_PerIoContextArr.size();i++){delete m_PerIoContextArr.at(i);}m_PerIoContextArr.clear();}_PER_IO_CONTEXT *GetNewIOContext(){_PER_IO_CONTEXT *p = new _PER_IO_CONTEXT;m_PerIoContextArr.push_back(p);return p;}void RemoveIOContext(_PER_IO_CONTEXT *pIoContext){assert(pIoContext != NULL);std::deque<PER_IO_CONTEXT *>::iterator iter = m_PerIoContextArr.begin();for (; iter != m_PerIoContextArr.end(); ++iter){if (*iter == pIoContext){iter = m_PerIoContextArr.erase(iter);break;}}}}PER_SOCKET_CONTEXT,*PPER_SOCKET_CONTEXT;class CIOCPModel;typedef struct  _tagThreadParameter_WORKER
{CIOCPModel *pIOCPModel;int			dwThreadNo;
}THREADPARAMS_WORKER, *PTHREADPARAMS_WORKER;class CIOCPModel
{
public:CIOCPModel();~CIOCPModel();public:bool start();void stop();bool LoadSocketLib();void unLoadSocketLib();std::string GetLocalIP();void setPort(const int &m_port);protected:bool _InitIOCPModel();bool _InitializeListenSocket();void _DeInitializeIOCPModel();bool _PostAccept(PER_IO_CONTEXT *PerContext);bool _PostRecv(PER_IO_CONTEXT *PerContext);bool _PostSend(PER_IO_CONTEXT *PerContext);bool _DoAccept(PER_SOCKET_CONTEXT *PerSocket, PER_IO_CONTEXT *PerContext);bool _DoRecv(PER_SOCKET_CONTEXT *PerSocket, PER_IO_CONTEXT *PerContext);void _AddToContextList(PER_SOCKET_CONTEXT *pSocketContext);void _RemoveContextList(PER_SOCKET_CONTEXT *pSocketContext);void _ClearContextList();bool _AssociateWithIOCP(PER_SOCKET_CONTEXT *pSocketContext);bool HandleError(PER_SOCKET_CONTEXT *pSocketContext, const DWORD &dwError);static DWORD __stdcall _WorkThread(LPVOID lpParam);int _GetNoOfProcessor();bool _IsSocketAlive(SOCKET s);
private:HANDLE m_hShutDownEvent;HANDLE m_hIOCompletionPort;HANDLE *m_WorkThread;int     m_nThreads;std::string  m_strIP;int			 m_port;CRITICAL_SECTION		m_cs;std::deque<PER_SOCKET_CONTEXT *>  m_ClientContextDeque;PER_SOCKET_CONTEXT		*pListenSocketContext;LPFN_ACCEPTEX                m_lpfnAcceptEx;LPFN_GETACCEPTEXSOCKADDRS    m_lpfnGetAcceptExSockAddrs;};#endif

相关文章:

《网络编程基础之完成端口模型》

【完成端口模型导读】完成端口模型&#xff0c;算得上是真正的异步网络IO模型吧&#xff0c;相对于其它网络IO模型&#xff0c;操作系统通知我们的时候&#xff0c;要么就是连接已经帮我建立好&#xff0c;客户端套接字帮我们准备好&#xff1b;要么就是数据已经接收完成&#…...

Axure PR 9 旋转效果 设计交互

大家好&#xff0c;我是大明同学。 这期内容&#xff0c;我们将学习Axure中的旋转效果设计与交互技巧。 旋转 创建旋转效果所需的元件 1.打开一个新的 RP 文件并在画布上打开 Page 1。 2.在元件库中拖出一个按钮元件。 创建交互 创建按钮交互状态 1.选中按钮元件&#xf…...

完美还是完成?把握好度,辨证看待

完美还是完成&#xff1f; 如果说之前这个答案有争议&#xff0c;那么现在&#xff0c;答案毋庸置疑 ■为什么完美大于完成 ●时间成本&#xff1a; 做事不仅要考虑结果&#xff0c;还要考虑时间和精力&#xff0c;要说十年磨一剑的确质量更好&#xff0c;但是现实没有那么多…...

C++的类Class

文章目录 一、C的struct和C的类的区别二、关于OOP三、举例&#xff1a;一个商品类CGoods四、构造函数和析构函数1、定义一个顺序栈2、用构造和析构代替s.init(5);和s.release();3、在不同内存区域构造对象4、深拷贝和浅拷贝5、构造函数和深拷贝的简单应用6、构造函数的初始化列…...

C++中的内存管理

学完了类与对象&#xff0c;这节我们来了解一下内存里的那些事 文章目录 一、C/C中的内存分布 1. 常量区&#xff08;代码段&#xff09; (Text Segment) 2. 静态区&#xff08;数据段&#xff09; (Data Segment) 3. 堆区 (Heap) 4. 栈区 (Stack) 5. 内存映射区域 (Memory-map…...

MySQL为什么默认引擎是InnoDB ?

大家好&#xff0c;我是锋哥。今天分享关于【MySQL为什么默认引擎是InnoDB &#xff1f;】面试题。希望对大家有帮助&#xff1b; MySQL为什么默认引擎是InnoDB &#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 MySQL 默认引擎是 InnoDB&#xff0c;主要…...

ComfyUI安装调用DeepSeek——DeepSeek多模态之图形模型安装问题解决(ComfyUI-Janus-Pro)

ComfyUI 的 Janus-Pro 节点&#xff0c;一个统一的多模态理解和生成框架。 试用&#xff1a; https://huggingface.co/spaces/deepseek-ai/Janus-1.3B https://huggingface.co/spaces/deepseek-ai/Janus-Pro-7B https://huggingface.co/spaces/deepseek-ai/JanusFlow-1.3B 安装…...

电脑要使用cuda需要进行什么配置

在电脑上使用CUDA&#xff08;NVIDIA的并行计算平台和API&#xff09;&#xff0c;需要进行以下配置和准备&#xff1a; 1. 检查NVIDIA显卡支持 确保你的电脑拥有支持CUDA的NVIDIA显卡。 可以在NVIDIA官方CUDA支持显卡列表中查看显卡型号是否支持CUDA。 2. 安装NVIDIA显卡驱动…...

利用Muduo库实现简单且健壮的Echo服务器

一、muduo网络库主要提供了两个类&#xff1a; TcpServer&#xff1a;用于编写服务器程序 TcpClient&#xff1a;用于编写客户端程序 二、三个重要的链接库&#xff1a; libmuduo_net、libmuduo_base、libpthread 三、muduo库底层就是epoll线程池&#xff0c;其好处是…...

Scratch 《像素战场》系列综合游戏:像素战场游戏Ⅰ~Ⅲ 介绍

资源下载 Scratch《像素战场》系列综合游戏合集&#xff1a;像素战场游戏Ⅰ~Ⅲ压缩包 https://download.csdn.net/download/leyang0910/90332765 游戏操作介绍 Scratch 《像素战场Ⅰ》操作规则&#xff1a; 这是一款与朋友一起玩的 1v1 游戏。先赢得6轮胜利&#xff01; WA…...

Android学习制作app(ESP8266-01S连接-简单制作)

一、理论 部分理论见arduino学习-CSDN博客和Android Studio安装配置_android studio gradle 配置-CSDN博客 以下直接上代码和效果视频&#xff0c;esp01S的收发硬件代码目前没有分享&#xff0c;但是可以通过另一个手机网络调试助手进行模拟。也可以直接根据我的代码进行改动…...

三甲医院大型生信服务器多配置方案剖析与应用(2024版)

一、引言 1.1 研究背景与意义 在当今数智化时代&#xff0c;生物信息学作为一门融合生物学、计算机科学和信息技术的交叉学科&#xff0c;在三甲医院的科研和临床应用中占据着举足轻重的地位。随着高通量测序技术、医学影像技术等的飞速发展&#xff0c;生物医学数据呈爆发式…...

【Unity3D】实现横版2D游戏——单向平台(简易版)

目录 问题 项目Demo直接使用免费资源&#xff1a;Hero Knight - Pixel Art &#xff08;Asset Store搜索&#xff09; 打开Demo场景&#xff0c;进行如下修改&#xff0c;注意Tag是自定义标签SingleDirCollider using System.Collections; using System.Collections.Generic;…...

大白话讲清楚embedding原理

Embedding&#xff08;嵌入&#xff09;是一种将高维数据&#xff08;如单词、句子、图像等&#xff09;映射到低维连续向量的技术&#xff0c;其核心目的是通过向量表示捕捉数据之间的语义或特征关系。以下从原理、方法和应用三个方面详细解释Embedding的工作原理。 一、Embe…...

电脑优化大师-解决电脑卡顿问题

我们常常会遇到电脑运行缓慢、网速卡顿的情况&#xff0c;但又不知道是哪个程序在占用过多资源。这时候&#xff0c;一款能够实时监控网络和系统状态的工具就显得尤为重要了。今天&#xff0c;就来给大家介绍一款小巧实用的监控工具「TrafficMonitor」。 「TrafficMonitor 」是…...

el-table组件样式如何二次修改?

文章目录 前言一、去除全选框按钮样式二、表头颜色的修改 前言 ElementUI中的组件el-table表格组件提供了丰富的样式&#xff0c;有一个全选框的el-table组件&#xff0c;提供了全选框和多选。 一、去除全选框按钮样式 原本默认是有全选框的。假如有一些开发者&#xff0c;因…...

java练习(1)

两数之和&#xff08;题目来自力扣&#xff09; 给定一个整数数组 nums 和一个整数目标值 target&#xff0c;请你在该数组中找出 和为目标值 target 的那 两个 整数&#xff0c;并返回它们的数组下标。 你可以假设每种输入只会对应一个答案&#xff0c;并且你不能使用两次相…...

UbuntuWindows双系统安装

做系统盘&#xff1a; Ubuntu20.04双系统安装详解&#xff08;内容详细&#xff0c;一文通关&#xff01;&#xff09;_ubuntu 20.04-CSDN博客 ubuntu系统调整大小&#xff1a; 调整指南&#xff1a; 虚拟机中的Ubuntu扩容及重新分区方法_ubuntu重新分配磁盘空间-CSDN博客 …...

DeepSeek大模型技术深度解析:揭开Transformer架构的神秘面纱

摘要 DeepSeek大模型由北京深度求索人工智能基础技术研究有限公司开发&#xff0c;基于Transformer架构&#xff0c;具备卓越的自然语言理解和生成能力。该模型能够高效处理智能对话、文本生成和语义理解等复杂任务&#xff0c;标志着人工智能在自然语言处理领域的重大进展。 关…...

MusicFree-开源的第三方音乐在线播放和下载工具, 支持歌单导入[对标落雪音乐]

MusicFree 链接&#xff1a;https://pan.xunlei.com/s/VOI0RrVLTTWE9kkpt0U7ofGBA1?pwd4ei6#...

脑机新手指南(八):OpenBCI_GUI:从环境搭建到数据可视化(下)

一、数据处理与分析实战 &#xff08;一&#xff09;实时滤波与参数调整 基础滤波操作 60Hz 工频滤波&#xff1a;勾选界面右侧 “60Hz” 复选框&#xff0c;可有效抑制电网干扰&#xff08;适用于北美地区&#xff0c;欧洲用户可调整为 50Hz&#xff09;。 平滑处理&…...

React Native 开发环境搭建(全平台详解)

React Native 开发环境搭建&#xff08;全平台详解&#xff09; 在开始使用 React Native 开发移动应用之前&#xff0c;正确设置开发环境是至关重要的一步。本文将为你提供一份全面的指南&#xff0c;涵盖 macOS 和 Windows 平台的配置步骤&#xff0c;如何在 Android 和 iOS…...

逻辑回归:给不确定性划界的分类大师

想象你是一名医生。面对患者的检查报告&#xff08;肿瘤大小、血液指标&#xff09;&#xff0c;你需要做出一个**决定性判断**&#xff1a;恶性还是良性&#xff1f;这种“非黑即白”的抉择&#xff0c;正是**逻辑回归&#xff08;Logistic Regression&#xff09;** 的战场&a…...

在鸿蒙HarmonyOS 5中实现抖音风格的点赞功能

下面我将详细介绍如何使用HarmonyOS SDK在HarmonyOS 5中实现类似抖音的点赞功能&#xff0c;包括动画效果、数据同步和交互优化。 1. 基础点赞功能实现 1.1 创建数据模型 // VideoModel.ets export class VideoModel {id: string "";title: string ""…...

相机Camera日志实例分析之二:相机Camx【专业模式开启直方图拍照】单帧流程日志详解

【关注我&#xff0c;后续持续新增专题博文&#xff0c;谢谢&#xff01;&#xff01;&#xff01;】 上一篇我们讲了&#xff1a; 这一篇我们开始讲&#xff1a; 目录 一、场景操作步骤 二、日志基础关键字分级如下 三、场景日志如下&#xff1a; 一、场景操作步骤 操作步…...

学习STC51单片机32(芯片为STC89C52RCRC)OLED显示屏2

每日一言 今天的每一份坚持&#xff0c;都是在为未来积攒底气。 案例&#xff1a;OLED显示一个A 这边观察到一个点&#xff0c;怎么雪花了就是都是乱七八糟的占满了屏幕。。 解释 &#xff1a; 如果代码里信号切换太快&#xff08;比如 SDA 刚变&#xff0c;SCL 立刻变&#…...

【7色560页】职场可视化逻辑图高级数据分析PPT模版

7种色调职场工作汇报PPT&#xff0c;橙蓝、黑红、红蓝、蓝橙灰、浅蓝、浅绿、深蓝七种色调模版 【7色560页】职场可视化逻辑图高级数据分析PPT模版&#xff1a;职场可视化逻辑图分析PPT模版https://pan.quark.cn/s/78aeabbd92d1...

【Android】Android 开发 ADB 常用指令

查看当前连接的设备 adb devices 连接设备 adb connect 设备IP 断开已连接的设备 adb disconnect 设备IP 安装应用 adb install 安装包的路径 卸载应用 adb uninstall 应用包名 查看已安装的应用包名 adb shell pm list packages 查看已安装的第三方应用包名 adb shell pm list…...

Spring Security 认证流程——补充

一、认证流程概述 Spring Security 的认证流程基于 过滤器链&#xff08;Filter Chain&#xff09;&#xff0c;核心组件包括 UsernamePasswordAuthenticationFilter、AuthenticationManager、UserDetailsService 等。整个流程可分为以下步骤&#xff1a; 用户提交登录请求拦…...

五子棋测试用例

一.项目背景 1.1 项目简介 传统棋类文化的推广 五子棋是一种古老的棋类游戏&#xff0c;有着深厚的文化底蕴。通过将五子棋制作成网页游戏&#xff0c;可以让更多的人了解和接触到这一传统棋类文化。无论是国内还是国外的玩家&#xff0c;都可以通过网页五子棋感受到东方棋类…...