AcceptEx获取远程ip和端口

iocp模型的tcp服务端若采用AcceptEx接受连接,

在有客户端连接后要获取客户端的ip和端口信息

流程:

AcceptEx

在工作线程收到客户端连接时

复制listensocket的信息到新客户端的socket

setsockopt( pOverlapped->hSocket, SOL_SOCKET, SO_UPDATE_ACCEPT_CONTEXT, (char *)&m_hSocket, sizeof(m_hSocket) );

然后就可以用getpeername获取远程ip和端口了.

转载于:https://www.cnblogs.com/Athrob/archive/2012/12/25/2832791.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
IOCP(Input/Output Completion Ports)是一种Windows系统下的高性能网络编程方式,通过异步I/O模型和多线程技术来提升服务器的吞吐量和性能。下面是一个简单的IOCP服务器代码示例: ```c++ #include <iostream> #include <Winsock2.h> #include <Windows.h> #pragma comment(lib, "ws2_32.lib") #define MAX_BUFFER_SIZE 4096 #define SERVER_PORT 8080 struct ClientContext { OVERLAPPED overlapped; SOCKET socket; CHAR buffer[MAX_BUFFER_SIZE]; WSABUF wsaBuf; }; DWORD WINAPI ServerWorkerThread(LPVOID lpParam) { HANDLE iocp = (HANDLE)lpParam; DWORD numBytes = 0; ULONG_PTR completionKey = 0; LPOVERLAPPED overlapped = NULL; ClientContext* clientContext = NULL; while (TRUE) { if (!GetQueuedCompletionStatus(iocp, &numBytes, &completionKey, &overlapped, INFINITE)) { std::cout << "IOCP Error: " << GetLastError() << std::endl; continue; } clientContext = CONTAINING_RECORD(overlapped, ClientContext, overlapped); if (numBytes == 0) { closesocket(clientContext->socket); delete clientContext; continue; } std::cout << "Received data: " << clientContext->buffer << std::endl; // Process client request // Send response to client // Prepare for next I/O operation ZeroMemory(&(clientContext->overlapped), sizeof(OVERLAPPED)); clientContext->wsaBuf.buf = clientContext->buffer; clientContext->wsaBuf.len = MAX_BUFFER_SIZE; DWORD flags = 0; DWORD recvBytes = 0; int ret = WSARecv(clientContext->socket, &(clientContext->wsaBuf), 1, &recvBytes, &flags, &(clientContext->overlapped), NULL); if (ret == SOCKET_ERROR && WSAGetLastError() != WSA_IO_PENDING) { std::cout << "WSARecv Error: " << WSAGetLastError() << std::endl; closesocket(clientContext->socket); delete clientContext; } } return 0; } int main() { WSADATA wsaData; if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) { std::cout << "Failed to initialize Winsock" << std::endl; return -1; } // Create socket SOCKET listenSocket = socket(AF_INET, SOCK_STREAM, 0); if (listenSocket == INVALID_SOCKET) { std::cout << "Failed to create socket" << std::endl; return -1; } // Bind socket SOCKADDR_IN serverAddr; serverAddr.sin_family = AF_INET; serverAddr.sin_port = htons(SERVER_PORT); serverAddr.sin_addr.S_un.S_addr = INADDR_ANY; if (bind(listenSocket, (SOCKADDR*)&serverAddr, sizeof(serverAddr)) == SOCKET_ERROR) { std::cout << "Failed to bind socket" << std::endl; return -1; } // Listen if (listen(listenSocket, SOMAXCONN) == SOCKET_ERROR) { std::cout << "Failed to listen" << std::endl; return -1; } // Create IOCP HANDLE iocp = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, 0); if (iocp == NULL) { std::cout << "Failed to create IOCP" << std::endl; return -1; } SYSTEM_INFO systemInfo; GetSystemInfo(&systemInfo); int numThreads = systemInfo.dwNumberOfProcessors * 2; for (int i = 0; i < numThreads; i++) { HANDLE threadHandle = CreateThread(NULL, 0, ServerWorkerThread, iocp, 0, NULL); CloseHandle(threadHandle); } // Associate listen socket with IOCP CreateIoCompletionPort((HANDLE)listenSocket, iocp, 0, 0); while (TRUE) { // Accept incoming client connections SOCKADDR_IN clientAddr; int addrLen = sizeof(clientAddr); SOCKET clientSocket = accept(listenSocket, (SOCKADDR*)&clientAddr, &addrLen); if (clientSocket == INVALID_SOCKET) { std::cout << "Failed to accept client connection" << std::endl; continue; } // Register client socket with IOCP ClientContext* clientContext = new ClientContext(); clientContext->socket = clientSocket; ZeroMemory(&(clientContext->overlapped), sizeof(OVERLAPPED)); clientContext->wsaBuf.buf = clientContext->buffer; clientContext->wsaBuf.len = MAX_BUFFER_SIZE; DWORD flags = 0; DWORD recvBytes = 0; int ret = WSARecv(clientSocket, &(clientContext->wsaBuf), 1, &recvBytes, &flags, &(clientContext->overlapped), NULL); if (ret == SOCKET_ERROR && WSAGetLastError() != WSA_IO_PENDING) { std::cout << "WSARecv Error: " << WSAGetLastError() << std::endl; closesocket(clientSocket); delete clientContext; continue; } } closesocket(listenSocket); WSACleanup(); return 0; } ``` 以上是一个简单的IOCP服务器代码示例,通过IOCP模型和多线程技术实现异步高性能的网络通信。服务器在循环中等待客户端连接请求,并通过AcceptEx函数接受客户端连接,并将相关的客户端Socket与IOCP关联。每个线程通过WaitForMultipleObjects函数等待来自客户端的I/O完成事件,并进行相应的处理。对于已连接的客户端,通过WSARecv函数接收客户端发送的数据,并根据具体业务逻辑进行处理,然后通过WSASend函数发送响应给客户端,最后继续等待下一个I/O操作的完成。这样的方式可以充分利用系统资源,提高服务器的并发性能和吞吐量。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值