SOCKET编程进阶之Overlapped IO完成例程模型

SOCKET编程进阶之Overlapped IO完成例程模型
完成例程模型相比与事件通知模型有个很大的优点就是不再受64个消息的限制,一个线程可以同时管理成百上千个socket连接,且保持较高的性能。 
完成例程相比与完成端口较为逊色,因为它的性能不能随着系统CPU数量的增长而线程增长,不过在我看来已经很强了,呵呵~! 
说白了,这些连接都是由系统来帮你管理的。你只需做的一件事就是:开启一个线程来accept进来的连接,剩下的工作交由系统来处理。而你,则需要提供给系统一个回调函数,发生新的网络事件的时候系统将执行这个函数: 
procedure WorkerRoutine( const dwError, cbTransferred : DWORD; const lpOverlapped : LPWSAOVERLAPPED; const dwFlags : DWORD ); stdcall; 
然后告诉系统用WorkerRoutine函数处理接收到的数据: 
WSARecv( m_socket, @FBuf, 1, dwTemp, dwFlag, @m_overlap, WorkerRoutine ); 
然后......没有什么然后了,系统什么都给你做了! 


源码--------------------------------------------------------------------------------------------- 
#pragma comment(lib,"ws2_32.lib") 
#include <winsock2.h> 
#include <stdio.h> 
#define DATA_BUFSIZE 1024        // 接收缓冲区大小 
#define MAXSESSION 10000        // 最大连接数 
typedef struct _SOCKET_INFORMATION { 
   OVERLAPPED Overlapped; 
   SOCKET Socket; 
   WSABUF DataBuf; 
   DWORD BytesSEND; 
   DWORD BytesRECV; 
} SOCKET_INFORMATION, * LPSOCKET_INFORMATION; 


SOCKET        ListenSocket = INVALID_SOCKET; 
DWORD   Flags = 0;                                                                // WSARecv的参数 
void CALLBACK WorkerRoutine(DWORD Error, DWORD BytesTransferred,LPWSAOVERLAPPED Overlapped, DWORD InFlags); 
DWORD WINAPI AcceptThread(LPVOID lpParameter) 

        WSADATA wsaData; 
        WSAStartup(MAKEWORD(2,2),&wsaData); 
        ListenSocket = WSASocket(AF_INET,SOCK_STREAM,IPPROTO_TCP,NULL,NULL,WSA_FLAG_OVERLAPPED); 
        SOCKADDR_IN ServerAddr; 
        ServerAddr.sin_family = AF_INET; 
        ServerAddr.sin_addr.S_un.S_addr = htonl(INADDR_ANY); 
        ServerAddr.sin_port = htons(1234); 
        bind(ListenSocket,(LPSOCKADDR)&ServerAddr,sizeof(ServerAddr)); 
        listen(ListenSocket,100); 
        printf("listenning...\n"); 
        SOCKADDR_IN ClientAddr; 
        int addr_length=sizeof(ClientAddr); 
        while (TRUE) 
        { 
                LPSOCKET_INFORMATION  SI = new SOCKET_INFORMATION; 
                if ((SI->Socket = accept(ListenSocket,(SOCKADDR*)&ClientAddr, &addr_length)) != INVALID_SOCKET) 
                { 
                        printf("accept ip:%s port:%d\n",inet_ntoa(ClientAddr.sin_addr),ClientAddr.sin_port); 
                        memset(&SI->Overlapped,0,sizeof(WSAOVERLAPPED)); 
                        SI->DataBuf.buf = new char[DATA_BUFSIZE]; 
                        SI->DataBuf.len = DATA_BUFSIZE; 
                        memset(SI->DataBuf.buf,0,DATA_BUFSIZE); 
                        if(WSARecv(SI->Socket, &SI->DataBuf, 1, &SI->BytesRECV, &Flags, &SI->Overlapped, WorkerRoutine) == SOCKET_ERROR) 
                        { 
                                int err = WSAGetLastError(); 
                                if(WSAGetLastError() != WSA_IO_PENDING) 
                                { 
                                        printf("disconnect\n"); 
                                        closesocket(SI->Socket); 
                                        delete [] SI->DataBuf.buf; 
                                        delete SI; 
                                        continue; 
                                } 
                        } 
                } 
                 
        } 
return FALSE; 

void CALLBACK WorkerRoutine(DWORD Error, DWORD BytesTransferred, LPWSAOVERLAPPED Overlapped, DWORD InFlags) 

        LPSOCKET_INFORMATION SI = (LPSOCKET_INFORMATION)Overlapped; 
        if (Error != 0 || BytesTransferred == 0) 
        { 
                printf("disconnect\n"); 
                closesocket(SI->Socket); 
                delete [] SI->DataBuf.buf; 
                delete SI; 
                return; 
        } 
        //使用数据 
        printf("call back:%s\n",SI->DataBuf.buf); 
        memset(SI->DataBuf.buf,0,DATA_BUFSIZE); 
         
        if(WSARecv(SI->Socket, &SI->DataBuf, 1, &SI->BytesRECV, &Flags, &SI->Overlapped, WorkerRoutine) == SOCKET_ERROR) 
        { 
                int err = WSAGetLastError(); 
                if(WSAGetLastError() != WSA_IO_PENDING) 
                { 
                        printf("disconnect\n"); 
                        closesocket(SI->Socket); 
                        delete [] SI->DataBuf.buf; 
                        delete SI; 
                        return; 
                } 
        } 

void main()   

        HANDLE hThreads = CreateThread(NULL, 0, AcceptThread, NULL, NULL, NULL); 
           
        WaitForSingleObject(hThreads,INFINITE); 
        printf("exit\n"); 
        CloseHandle(hThreads); 

 
唉~花了好长时间才搞定了Overlapped I\O ,不过这还不是winsock编程的最高境界;
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
重叠IO模型OverLapped完成例程模型WSACompletionRoutineServer VS2010 基础入门 客户端与服务器端 客户端向服务器端发送数据 可接收多个客户端 #include #include #pragma comment (lib, "ws2_32.lib") #define PORT 8088 #define MSG_SIZE 1024 SOCKET g_sConnect; bool g_bConnect = false; typedef struct { WSAOVERLAPPED overLap; WSABUF wsaBuf; char chMsg[MSG_SIZE]; DWORD nRecvNum; DWORD nFlags; SOCKET sClient; }PRE_IO_OPERATION_DATA, *LP_PER_IO_OPERATION_DATA; void CALLBACK CompletionRoutine(DWORD dwError, DWORD dwTrans, LPWSAOVERLAPPED lpOverlap, DWORD nFlags); DWORD WINAPI workThread(LPVOID lp) { LP_PER_IO_OPERATION_DATA lpData; while(TRUE) { if (g_bConnect) // 有新的连接 { // 为lpData分配空间并初始化 lpData = (LP_PER_IO_OPERATION_DATA)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(PRE_IO_OPERATION_DATA)); lpData->wsaBuf.len = MSG_SIZE; lpData->wsaBuf.buf = lpData->chMsg; lpData->sClient = g_sConnect; WSARecv(lpData->sClient, &lpData->wsaBuf, 1, &lpData->nRecvNum, &lpData->nFlags, &lpData->overLap, CompletionRoutine); g_bConnect = false; // 处理完毕 } SleepEx(1000, TRUE); } return 0; } // 系统在WSARecv收到信息后,自动调用此函数,并传入参数--回调函数 void CALLBACK CompletionRoutine(DWORD dwError, DWORD dwTrans, LPWSAOVERLAPPED lpOverlap, DWORD nFlags) { LP_PER_IO_OPERATION_DATA lpData = (LP_PER_IO_OPERATION_DATA)lpOverlap; if (0 != dwError) // 接收失败 { printf("Socket %d Close!\n", lpData->sClient); closesocket(lpData->sClient); HeapFree(GetProcessHeap(), 0, lpData); } else // 接收成功 { lpData->chMsg[dwTrans] = '\0'; send(lpData->sClient, lpData->chMsg, dwTrans, 0); printf("Socket:%d MSG: %s \n", lpData->sClient, lpData->chMsg); memset(&lpData->overLap, 0, sizeof(WSAOVERLAPPED)); lpData->wsaBuf.len = MSG_SIZE; lpData->wsaBuf.buf = lpData->chMsg; // 继续接收来自客户端的数据 实现 WSARecv与CompletionRoutine循环 WSARecv(lpData->sClient, &lpData->wsaBuf,1, &lpData->nRecvNum, &lpData->nFlags, &lpData->overLap, CompletionRoutine); } } int main() { WSADATA wsaData; WSAStartup(0x0202, &wsaData); SOCKET sListen; sListen = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); sockaddr_in addrListen; addrListen.sin_family = AF_INET; addrListen.sin_port = htons(PORT); addrListen.sin_addr.S_un.S_addr = htonl(ADDR_ANY); int nErrorCode = 0; nErrorCode = bind(sListen, (sockaddr*)&addrListen, sizeof(sockaddr)); nErrorCode = listen(sListen, 5); DWORD nThreadID; CreateThread(NULL, 0, workThread, NULL, 0, &nThreadID); sockaddr_in addrConnect; int nAddrLen = sizeof(sockaddr_in); printf("Server Started!\n"); while(TRUE) { g_sConnect= accept(sListen, (sockaddr*)&addrConnect, &nAddrLen); if (INVALID_SOCKET == g_sConnect) { return -1; } g_bConnect = true; // 连接成功 printf("Accept Client :%s -- PORT:%d\n", inet_ntoa(addrConnect.sin_addr), htons(addrConnect.sin_port)); } return 0; }
基于完成端口模型socket编程设计思路如下: 1. 创建完成端口:使用CreateIoCompletionPort函数创建一个完成端口对象,该对象用于管理异步I/O操作的完成状态。 2. 创建套接字:使用socket函数创建一个套接字,用于与客户端进行通信。 3. 绑定套接字到完成端口:使用CreateIoCompletionPort函数将套接字绑定到完成端口对象上,以便将套接字的I/O操作与完成端口关联起来。 4. 接收客户端连接:使用accept函数接受客户端的连接请求,并将返回的套接字与完成端口对象关联起来。 5. 创建工作线程:使用CreateThread函数创建多个工作线程,每个线程都会调用GetQueuedCompletionStatus函数来等待I/O操作完成。 6. 异步I/O操作:在工作线程中,使用WSARecv和WSASend函数进行异步的接收和发送操作。当操作完成时,将结果信息和相关的套接字句柄放入完成端口队列中。 7. 处理完成的I/O操作:在工作线程中,使用GetQueuedCompletionStatus函数从完成端口队列中获取已完成的I/O操作信息。根据操作类型进行相应的处理,例如处理接收到的数据或发送完成的通知。 8. 关闭套接字和完成端口:在程序结束时,使用closesocket函数关闭套接字,并使用CloseHandle函数关闭完成端口对象。 以下是一个基于完成端口模型socket编程的示例代码: ```c // 创建完成端口 HANDLE hPort = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, 0); if (hPort == 0) { int error = GetLastError(); printf("创建完成端口失败:%d\n", error); return 0; } // 创建套接字 SOCKET socketServer = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (socketServer == INVALID_SOCKET) { int error = WSAGetLastError(); printf("创建套接字失败:%d\n", error); return 0; } // 绑定套接字到完成端口 HANDLE hPort1 = CreateIoCompletionPort((HANDLE)socketServer, hPort, 0, 0); if (hPort1 != hPort) { int error = GetLastError(); printf("绑定套接字到完成端口失败:%d\n", error); closesocket(socketServer); return 0; } // 接收客户端连接 SOCKADDR_IN serverAddr; serverAddr.sin_family = AF_INET; serverAddr.sin_port = htons(12345); serverAddr.sin_addr.s_addr = INADDR_ANY; if (bind(socketServer, (SOCKADDR*)&serverAddr, sizeof(serverAddr)) == SOCKET_ERROR) { int error = WSAGetLastError(); printf("绑定套接字失败:%d\n", error); closesocket(socketServer); return 0; } if (listen(socketServer, SOMAXCONN) == SOCKET_ERROR) { int error = WSAGetLastError(); printf("监听套接字失败:%d\n", error); closesocket(socketServer); return 0; } // 创建工作线程 DWORD WINAPI WorkerThread(LPVOID lpParam) { HANDLE hPort = (HANDLE)lpParam; DWORD bytesTransferred; ULONG_PTR completionKey; LPOVERLAPPED overlapped; while (GetQueuedCompletionStatus(hPort, &bytesTransferred, &completionKey, &overlapped, INFINITE)) { // 处理完成的I/O操作 // ... } return 0; } for (int i = 0; i < NUM_THREADS; i++) { HANDLE hThread = CreateThread(NULL, 0, WorkerThread, hPort, 0, NULL); if (hThread == NULL) { int error = GetLastError(); printf("创建工作线程失败:%d\n", error); return 0; } } // 异步I/O操作 while (true) { SOCKET socketClient = accept(socketServer, NULL, NULL); if (socketClient == INVALID_SOCKET) { int error = WSAGetLastError(); printf("接受客户端连接失败:%d\n", error); break; } // 创建重叠结构体 // ... // 异步接收数据 WSARecv(socketClient, &wsaBuf, 1, &bytesReceived, &flags, overlapped, NULL); // 异步发送数据 WSASend(socketClient, &wsaBuf, 1, &bytesSent, flags, overlapped, NULL); } // 关闭套接字和完成端口 closesocket(socketServer); CloseHandle(hPort); ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值