SOCKET编程进阶之Overlapped I/O完成例程模型

SOCKET编程进阶之Overlapped I/O完成例程模型

原文地址:http://blog.csdn.net/echoff/archive/2007/09/23/1797319.aspx

完成例程模型相比与事件通知模型有个很大的优点就是不再受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 ); 
然后......没有什么然后了,系统什么都给你做了! 

  1. #pragma comment(lib,"ws2_32.lib") 
  2. #include <winsock2.h> 
  3. #include <stdio.h> 
  4. #define DATA_BUFSIZE 1024        // 接收缓冲区大小 
  5. #define MAXSESSION 10000        // 最大连接数 
  6. typedef struct _SOCKET_INFORMATION { 
  7.    OVERLAPPED Overlapped; 
  8.    SOCKET Socket; 
  9.    WSABUF DataBuf; 
  10.    DWORD BytesSEND; 
  11.    DWORD BytesRECV; 
  12. } SOCKET_INFORMATION, * LPSOCKET_INFORMATION; 
  13. SOCKET        ListenSocket = INVALID_SOCKET; 
  14. DWORD   Flags = 0;                                                                // WSARecv的参数 
  15. void CALLBACK WorkerRoutine(DWORD Error, DWORD BytesTransferred,LPWSAOVERLAPPED Overlapped, DWORD InFlags); 
  16. DWORD WINAPI AcceptThread(LPVOID lpParameter) 
  17.         WSADATA wsaData; 
  18.         WSAStartup(MAKEWORD(2,2),&wsaData); 
  19.         ListenSocket = WSASocket(AF_INET,SOCK_STREAM,IPPROTO_TCP,NULL,NULL,WSA_FLAG_OVERLAPPED); 
  20.         SOCKADDR_IN ServerAddr; 
  21.         ServerAddr.sin_family = AF_INET; 
  22.         ServerAddr.sin_addr.S_un.S_addr = htonl(INADDR_ANY); 
  23.         ServerAddr.sin_port = htons(1234); 
  24.         bind(ListenSocket,(LPSOCKADDR)&ServerAddr,sizeof(ServerAddr)); 
  25.         listen(ListenSocket,100); 
  26.         printf("listenning.../n"); 
  27.         SOCKADDR_IN ClientAddr; 
  28.         int addr_length=sizeof(ClientAddr); 
  29.         while (TRUE) 
  30.         { 
  31.                 LPSOCKET_INFORMATION  SI = new SOCKET_INFORMATION; 
  32.                 if ((SI->Socket = accept(ListenSocket,(SOCKADDR*)&ClientAddr, &addr_length)) != INVALID_SOCKET) 
  33.                 { 
  34.                         printf("accept ip:%s port:%d/n",inet_ntoa(ClientAddr.sin_addr),ClientAddr.sin_port); 
  35.                         memset(&SI->Overlapped,0,sizeof(WSAOVERLAPPED)); 
  36.                         SI->DataBuf.buf = new char[DATA_BUFSIZE]; 
  37.                         SI->DataBuf.len = DATA_BUFSIZE; 
  38.                         memset(SI->DataBuf.buf,0,DATA_BUFSIZE); 
  39.                         if(WSARecv(SI->Socket, &SI->DataBuf, 1, &SI->BytesRECV, &Flags, &SI->Overlapped, WorkerRoutine) == SOCKET_ERROR) 
  40.                         { 
  41.                                 int err = WSAGetLastError(); 
  42.                                 if(WSAGetLastError() != WSA_IO_PENDING) 
  43.                                 { 
  44.                                         printf("disconnect/n"); 
  45.                                         closesocket(SI->Socket); 
  46.                                         delete [] SI->DataBuf.buf; 
  47.                                         delete SI; 
  48.                                         continue
  49.                                 } 
  50.                         } 
  51.                 } 
  52.                  
  53.         } 
  54. return FALSE; 
  55. void CALLBACK WorkerRoutine(DWORD Error, DWORD BytesTransferred, LPWSAOVERLAPPED Overlapped, DWORD InFlags) 
  56.         LPSOCKET_INFORMATION SI = (LPSOCKET_INFORMATION)Overlapped; 
  57.         if (Error != 0 || BytesTransferred == 0) 
  58.         { 
  59.                 printf("disconnect/n"); 
  60.                 closesocket(SI->Socket); 
  61.                 delete [] SI->DataBuf.buf; 
  62.                 delete SI; 
  63.                 return
  64.         } 
  65.         //使用数据 
  66.         printf("call back:%s/n",SI->DataBuf.buf); 
  67.         memset(SI->DataBuf.buf,0,DATA_BUFSIZE); 
  68.          
  69.         if(WSARecv(SI->Socket, &SI->DataBuf, 1, &SI->BytesRECV, &Flags, &SI->Overlapped, WorkerRoutine) == SOCKET_ERROR) 
  70.         { 
  71.                 int err = WSAGetLastError(); 
  72.                 if(WSAGetLastError() != WSA_IO_PENDING) 
  73.                 { 
  74.                         printf("disconnect/n"); 
  75.                         closesocket(SI->Socket); 
  76.                         delete [] SI->DataBuf.buf; 
  77.                         delete SI; 
  78.                         return
  79.                 } 
  80.         } 
  81. void main()   
  82.         HANDLE hThreads = CreateThread(NULL, 0, AcceptThread, NULL, NULL, NULL); 
  83.            
  84.         WaitForSingleObject(hThreads,INFINITE); 
  85.         printf("exit/n"); 
  86.         CloseHandle(hThreads); 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值