IOCP簡單實現代碼

來源:互聯網
上載者:User

 簡單實現,適合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. }

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.