IOCP简单实现代码

52 篇文章 0 订阅

 简单实现,适合IOCP入门
参考:《WINDOWS网络与通信程序设计》

 

  1. /******************************************************************
  2. *
  3. * Copyright (c) 2008, xxxxx有限公司
  4. * All rights reserved.
  5. *
  6. * 文件名称:IOCPHeader.h
  7. * 摘   要: IOCP定义文件
  8. *
  9. * 当前版本:1.0
  10. * 作    者:吴会然
  11. * 完成日期:2008-9-16
  12. *
  13. * 取代版本:
  14. * 原  作者:
  15. * 完成日期:
  16. *
  17. ******************************************************************/
  18. #ifndef _IOCPHEADER_H_20080916_
  19. #define _IOCPHEADER_H_20080916_
  20. #include <WINSOCK2.H>
  21. #include <windows.h>
  22. #define BUFFER_SIZE 1024
  23. /******************************************************************
  24. * per_handle 数据
  25. *******************************************************************/
  26. typedef struct _PER_HANDLE_DATA 
  27. {
  28.     SOCKET      s;      // 对应的套接字句柄
  29.     sockaddr_in addr;   // 对方的地址
  30. }PER_HANDLE_DATA, *PPER_HANDLE_DATA;
  31. /******************************************************************
  32. * per_io 数据
  33. *******************************************************************/
  34. typedef struct _PER_IO_DATA
  35. {
  36.     OVERLAPPED  ol;                 // 重叠结构
  37.     char        buf[BUFFER_SIZE];   // 数据缓冲区
  38.     int         nOperationType;     // 操作类型
  39. #define OP_READ   1
  40. #define OP_WRITE  2
  41. #define OP_ACCEPT 3
  42. }PER_IO_DATA, *PPER_IO_DATA;
  43. #endif
  44. /******************************************************************
  45. *
  46. * Copyright (c) 2008, xxxxx有限公司
  47. * All rights reserved.
  48. *
  49. * 文件名称:main.cpp
  50. * 摘   要: iocp demo 
  51. *
  52. * 当前版本:1.0
  53. * 作    者:吴会然
  54. * 完成日期:2008-9-16
  55. *
  56. * 取代版本:
  57. * 原  作者:
  58. * 完成日期:
  59. *
  60. ******************************************************************/
  61. #include <iostream>
  62. #include <string>
  63. #include "IOCPHeader.h"
  64. using namespace std;
  65. DWORD WINAPI ServerThread( LPVOID lpParam );
  66. int main( int argc, char *argv[] )
  67. {
  68.     //  
  69.     WSADATA wsaData;
  70.     if( 0 != WSAStartup( MAKEWORD( 2, 2 ), &wsaData ) )
  71.     {
  72.         printf( "Using %s (Status:%s)/n", wsaData.szDescription, wsaData.szSystemStatus );
  73.         printf( "with API versions: %d.%d to %d.%d"
  74.                  LOBYTE( wsaData.wVersion), HIBYTE( wsaData.wVersion ),
  75.                  LOBYTE( wsaData.wHighVersion), HIBYTE( wsaData.wHighVersion) );
  76.         return -1;
  77.     }
  78.     else
  79.     {
  80.         printf("Windows sockets 2.2 startup/n");
  81.     }
  82.     //
  83.     int nPort = 20055;
  84.     // 创建完成端口对象
  85.     // 创建工作线程处理完成端口对象的事件
  86.     HANDLE hIocp = ::CreateIoCompletionPort( INVALID_HANDLE_VALUE, 0, 0, 0 );
  87.     ::CreateThread( NULL, 0, ServerThread, (LPVOID)hIocp, 0, 0 );
  88.     // 创建监听套接字,绑定本地端口,开始监听
  89.     SOCKET sListen = ::socket( AF_INET, SOCK_STREAM, 0 );
  90.     SOCKADDR_IN addr;
  91.     addr.sin_family = AF_INET;
  92.     addr.sin_port = ::htons( nPort );
  93.     addr.sin_addr.S_un.S_addr = INADDR_ANY;
  94.     ::bind( sListen, (sockaddr *)&addr, sizeof( addr ) );
  95.     ::listen( sListen, 5 );
  96.     printf( "iocp demo start....../n" );
  97.     // 循环处理到来的请求
  98.     while ( TRUE )
  99.     {
  100.         // 等待接受未决的连接请求
  101.         SOCKADDR_IN saRemote;
  102.         int nRemoteLen = sizeof( saRemote );
  103.         SOCKET sRemote = ::accept( sListen, (sockaddr *)&saRemote, &nRemoteLen );
  104.         // 接受到新连接之后,为它创建一个per_handle数据,并将他们关联到完成端口对象
  105.         PPER_HANDLE_DATA pPerHandle = ( PPER_HANDLE_DATA )::GlobalAlloc( GPTR, sizeof( PPER_HANDLE_DATA ) );
  106.         if( pPerHandle == NULL )
  107.         {
  108.             break;
  109.         }
  110.         pPerHandle->s = sRemote;
  111.         memcpy( &pPerHandle->addr, &saRemote, nRemoteLen );
  112.         ::CreateIoCompletionPort( ( HANDLE)pPerHandle->s, hIocp, (DWORD)pPerHandle, 0 );
  113.         // 投递一个接受请求
  114.         PPER_IO_DATA pIoData = ( PPER_IO_DATA )::GlobalAlloc( GPTR, sizeof( PPER_IO_DATA ) );
  115.         if( pIoData == NULL )
  116.         {
  117.             break;
  118.         }
  119.         pIoData->nOperationType = OP_READ;
  120.         WSABUF buf;
  121.         buf.buf = pIoData->buf;
  122.         buf.len = BUFFER_SIZE;
  123.         
  124.         DWORD dwRecv = 0;
  125.         DWORD dwFlags = 0;
  126.         ::WSARecv( pPerHandle->s, &buf, 1, &dwRecv, &dwFlags, &pIoData->ol, NULL );
  127.     }
  128.     //
  129. ERROR_PROC:
  130.     WSACleanup();
  131.     //
  132.     return 0;
  133. }
  134. /******************************************************************
  135. * 函数介绍:处理完成端口对象事件的线程
  136. * 输入参数:
  137. * 输出参数:
  138. * 返回值  :
  139. *******************************************************************/
  140. DWORD WINAPI ServerThread( LPVOID lpParam )
  141. {
  142.     HANDLE hIocp = ( HANDLE )lpParam;
  143.     if( hIocp == NULL )
  144.     {
  145.         return -1;
  146.     }
  147.     DWORD dwTrans = 0;
  148.     PPER_HANDLE_DATA pPerHandle;
  149.     PPER_IO_DATA     pPerIo;
  150.     
  151.     while( TRUE )
  152.     {
  153.         // 在关联到此完成端口的所有套接字上等待I/O完成
  154.         BOOL bRet = ::GetQueuedCompletionStatus( hIocp, &dwTrans, (LPDWORD)&pPerHandle, (LPOVERLAPPED*)&pPerIo, WSA_INFINITE );
  155.         if( !bRet )     // 发生错误
  156.         {
  157.             ::closesocket( pPerHandle->s );
  158.             ::GlobalFree( pPerHandle );
  159.             ::GlobalFree( pPerIo );
  160.             cout << "error" << endl;
  161.             continue;
  162.         }
  163.         // 套接字被对方关闭
  164.         if( dwTrans == 0 && ( pPerIo->nOperationType == OP_READ || pPerIo->nOperationType == OP_WRITE ) )
  165.         {
  166.             ::closesocket( pPerHandle->s );
  167.             ::GlobalFree( pPerHandle );
  168.             ::GlobalFree( pPerIo );
  169.             cout << "client closed" << endl;
  170.             continue;
  171.         }
  172.         switch ( pPerIo->nOperationType )
  173.         {
  174.         case OP_READ:       // 完成一个接收请求
  175.             {
  176.                 pPerIo->buf[dwTrans] = '/0';
  177.                 printf( "%s/n", pPerIo->buf );
  178.                 // 继续投递接受操作
  179.                 WSABUF buf;
  180.                 buf.buf = pPerIo->buf;
  181.                 buf.len = BUFFER_SIZE;
  182.                 pPerIo->nOperationType = OP_READ;
  183.                 
  184.                 DWORD dwRecv = 0;
  185.                 DWORD dwFlags = 0;
  186.                 
  187.                 ::WSARecv( pPerHandle->s, &buf, 1, &dwRecv, &dwFlags, &pPerIo->ol, NULL );
  188.             }
  189.             break;
  190.         case OP_WRITE:
  191.         case OP_ACCEPT:
  192.             break;
  193.         }
  194.     }
  195.     return 0;
  196. }
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值