青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

kenlistian

厚積薄發. 勤為槳,思為帆

   :: 首頁 :: 新隨筆 ::  :: 聚合  :: 管理 ::
  73 隨筆 :: 4 文章 :: 22 評論 :: 0 Trackbacks

來自微軟的完成端口例子,就講解一下它的使用套路吧
反正編程這個玩意,只要用過,自然就知道什么回事,一次不會再看一次,學習這個玩意,無他,勤奮而已。
奢談效率等等,那只是孰能生巧上的功夫。
 

  這個例子是在console下的例子,算是一個echo服務器吧,
  跑起來后將在5150端口監聽,一旦有個端口連接上來,發個數據給服務端口,它就echo回數據給那個端口. 直到那個連接中斷.
 
 完成端口,其實理解成一個通道或管子就可以了,和管道也差不了多少,不過可以實現異步處理罷了,
 你這邊往管子里丟數據,通過GetQueuedCompletionStatus來查管子那頭出數據沒,出了就處理,這個管子就是通過一個自定義有點特殊的結構來寫入或讀出數據而已.
 那個完成端口,其實就相當是標識那個數據塊的句柄,

//下面請看例子
#include <winsock2.h>
#include <windows.h>
#include <stdio.h>

#define PORT 5150
#define DATA_BUFSIZE 8192

#pragma comment(lib, "Ws2_32")

typedef struct                        //這個玩意就是灌數據,取數據的一個自定義數據結構

                                              //和那個wm_data差不了多少,不過就是老要塞一個OverLapped結構,
{
   OVERLAPPED Overlapped;
   WSABUF DataBuf;
   CHAR Buffer[DATA_BUFSIZE];                    
   DWORD BytesSEND;                                 //發送字節數
   DWORD BytesRECV;                                
} PER_IO_OPERATION_DATA, * LPPER_IO_OPERATION_DATA;


typedef struct
{
   SOCKET Socket;
} PER_HANDLE_DATA, * LPPER_HANDLE_DATA;


DWORD WINAPI ServerWorkerThread(LPVOID CompletionPortID);


void main(void)
{
   SOCKADDR_IN InternetAddr;
   SOCKET Listen;
   SOCKET Accept;
   HANDLE CompletionPort;
   SYSTEM_INFO SystemInfo;
   LPPER_HANDLE_DATA PerHandleData;
   LPPER_IO_OPERATION_DATA PerIoData;
   int i;
   DWORD RecvBytes;
   DWORD Flags;
   DWORD ThreadID;
   WSADATA wsaData;
   DWORD Ret;

   if ((Ret = WSAStartup(0x0202, &wsaData)) != 0)
   {
      printf("WSAStartup failed with error %d\n", Ret);
      return;
   }

   //
   //完成端口的建立得搞2次,這是第一次調用,至于為什么?我問問你
   //
   if ((CompletionPort = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, 0)) == NULL)
   {
      printf( "CreateIoCompletionPort failed with error: %d\n", GetLastError());
      return;
   }
   //老套子api,不談也罷
   GetSystemInfo(&SystemInfo);
  
   //發現2個CPU,那就開個雙倍的線程跑吧
   for(i = 0; i < SystemInfo.dwNumberOfProcessors * 2; i++)
   {
      HANDLE ThreadHandle;
     
      //
      //完成端口掛到線程上面來了,就像管子把灌數據的和讀數據的兩頭都連上了,           
     //
      if ((ThreadHandle = CreateThread(NULL, 0, ServerWorkerThread, CompletionPort,
         0, &ThreadID)) == NULL)
      {
         printf("CreateThread() failed with error %d\n", GetLastError());
         return;
      }     
      CloseHandle(ThreadHandle);
   }

   //
   //啟動一個監聽socket ,以下都是長長的交代
   //
   if ((Listen = WSASocket(AF_INET, SOCK_STREAM, 0, NULL, 0,
      WSA_FLAG_OVERLAPPED)) == INVALID_SOCKET)
   {
      printf("WSASocket() failed with error %d\n", WSAGetLastError());
      return;
   }

   InternetAddr.sin_family = AF_INET;
   InternetAddr.sin_addr.s_addr = htonl(INADDR_ANY);
   InternetAddr.sin_port = htons(PORT);

   if (bind(Listen, (PSOCKADDR) &InternetAddr, sizeof(InternetAddr)) == SOCKET_ERROR)
   {
      printf("bind() failed with error %d\n", WSAGetLastError());
      return;
   }  

   if (listen(Listen, 5) == SOCKET_ERROR)
   {
      printf("listen() failed with error %d\n", WSAGetLastError());
      return;
   }

   //
   // 監聽端口打開,就開始在這里循環,一有socket連上,WSAAccept就創建一個socket,
   // 這個socket 又和完成端口聯上,
   //
   // 嘿嘿,完成端口第二次調用那個createxxx函數,為什么,留給人思考思考可能更深刻,
   // 反正這套路得來2次,
   // 完成端口completionport和accept socket掛起來了,
   //
   while(TRUE)
   {

    //主線程跑到這里就等啊等啊,但是線程卻開工了,
      if ((Accept = WSAAccept(Listen, NULL, NULL, NULL, 0)) == SOCKET_ERROR)
      {
         printf("WSAAccept() failed with error %d\n", WSAGetLastError());
         return;
      }
     
      if ((PerHandleData = (LPPER_HANDLE_DATA) GlobalAlloc(GPTR, sizeof(PER_HANDLE_DATA))) == NULL)
      {
         printf("GlobalAlloc() failed with error %d\n", GetLastError());
         return;
      }     
     
      PerHandleData->Socket = Accept;
     
      //
     //把這頭和完成端口completionPort連起來
     //就像你把漏斗接到管子口上,開始要灌數據了
     //
      if (CreateIoCompletionPort((HANDLE) Accept, CompletionPort, (DWORD) PerHandleData,
         0) == NULL)
      {
         printf("CreateIoCompletionPort failed with error %d\n", GetLastError());
         return;
      }
     
      //
      //清管子的數據結構,準備往里面灌數據
      //
      if ((PerIoData = (LPPER_IO_OPERATION_DATA) GlobalAlloc(GPTR,sizeof(PER_IO_OPERATION_DATA))) == NULL)
      {
         printf("GlobalAlloc() failed with error %d\n", GetLastError());
         return;
      }

      ZeroMemory(&(PerIoData->Overlapped), sizeof(OVERLAPPED));
      PerIoData->BytesSEND = 0;
      PerIoData->BytesRECV = 0;
      PerIoData->DataBuf.len = DATA_BUFSIZE;
      PerIoData->DataBuf.buf = PerIoData->Buffer;

      Flags = 0;
     
      //
      //  accept接到了數據,就放到PerIoData中,而perIoData又通過線程中的函數取出,
     //
      if (WSARecv(Accept, &(PerIoData->DataBuf), 1, &RecvBytes, &Flags,
         &(PerIoData->Overlapped), NULL) == SOCKET_ERROR)
      {
         if (WSAGetLastError() != ERROR_IO_PENDING)
         {
            printf("WSARecv() failed with error %d\n", WSAGetLastError());
            return;
         }
      }
   }
}

//
//線程一但調用,就老在里面循環,
// 注意,傳入的可是完成端口啊,就是靠它去取出管子中的數據
//
DWORD WINAPI ServerWorkerThread(LPVOID CompletionPortID)
{
   HANDLE CompletionPort = (HANDLE) CompletionPortID;
  
   DWORD BytesTransferred;
   LPOVERLAPPED Overlapped;
   LPPER_HANDLE_DATA PerHandleData;
   LPPER_IO_OPERATION_DATA PerIoData;        
   DWORD SendBytes, RecvBytes;
   DWORD Flags;
 
   while(TRUE)
   {
      //
      //在這里檢查完成端口部分的數據buf區,數據來了嗎?
      // 這個函數參數要看說明,
      // PerIoData 就是從管子流出來的數據,
      //PerHandleData 也是從管子里取出的,是何時塞進來的,
     //就是在建立第2次createIocompletionPort時
    // 

      if (GetQueuedCompletionStatus(CompletionPort, &BytesTransferred,
         (LPDWORD)&PerHandleData, (LPOVERLAPPED *) &PerIoData, INFINITE) == 0)
      {
         printf("GetQueuedCompletionStatus failed with error %d\n", GetLastError());
         return 0;
      }

      // 檢查數據傳送完了嗎
      if (BytesTransferred == 0)
      {
         printf("Closing socket %d\n", PerHandleData->Socket);

         if (closesocket(PerHandleData->Socket) == SOCKET_ERROR)
         {
            printf("closesocket() failed with error %d\n", WSAGetLastError());
            return 0;
         }

         GlobalFree(PerHandleData);
         GlobalFree(PerIoData);
         continue;
      }    
     //
    //看看管子里面有數據來了嗎?=0,那是剛收到數據
    //
      if (PerIoData->BytesRECV == 0)
      {
         PerIoData->BytesRECV = BytesTransferred;
         PerIoData->BytesSEND = 0;
      }
      else   //來了,
      {
         PerIoData->BytesSEND += BytesTransferred;
      }
  
      //
      // 數據沒發完?繼續send出去
     //
     if (PerIoData->BytesRECV > PerIoData->BytesSEND)
      {

         ZeroMemory(&(PerIoData->Overlapped), sizeof(OVERLAPPED)); //清0為發送準備
         PerIoData->DataBuf.buf = PerIoData->Buffer + PerIoData->BytesSEND;
         PerIoData->DataBuf.len = PerIoData->BytesRECV - PerIoData->BytesSEND;

       //1個字節一個字節發送發送數據出去
         if (WSASend(PerHandleData->Socket, &(PerIoData->DataBuf), 1, &SendBytes, 0,
            &(PerIoData->Overlapped), NULL) == SOCKET_ERROR)
         {
            if (WSAGetLastError() != ERROR_IO_PENDING)
            {
               printf("WSASend() failed with error %d\n", WSAGetLastError());
               return 0;
            }
         }
      }
      else
      {
         PerIoData->BytesRECV = 0;

         Flags = 0;
         ZeroMemory(&(PerIoData->Overlapped), sizeof(OVERLAPPED));

         PerIoData->DataBuf.len = DATA_BUFSIZE;
         PerIoData->DataBuf.buf = PerIoData->Buffer;

         if (WSARecv(PerHandleData->Socket, &(PerIoData->DataBuf), 1, &RecvBytes, &Flags,
            &(PerIoData->Overlapped), NULL) == SOCKET_ERROR)
         {
            if (WSAGetLastError() != ERROR_IO_PENDING)
            {
               printf("WSARecv() failed with error %d\n", WSAGetLastError());
               return 0;
            }
         }
      }
   }
}

posted on 2006-05-25 22:27 kenlistian 閱讀(2468) 評論(0)  編輯 收藏 引用
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲免费影院| 亚洲精品系列| 免费在线日韩av| 久久久久在线观看| 欧美一区二区三区免费在线看| 日韩一级精品视频在线观看| 亚洲美女在线视频| 亚洲视频播放| 午夜精品久久久久久久男人的天堂 | 亚洲欧美高清| 欧美一区二区视频网站| 久久久97精品| 免费看亚洲片| 亚洲精品在线观看免费| 亚洲一区二区三区在线播放| 午夜精品久久久久久久久久久 | 欧美r片在线| 亚洲欧洲综合另类| 亚洲精品在线观| 午夜精品一区二区三区四区 | 午夜精品影院| 蜜桃久久精品乱码一区二区| 欧美视频专区一二在线观看| 国产欧美一区二区三区视频| 亚洲狠狠婷婷| 欧美在线免费视频| 亚洲国产欧洲综合997久久| 中文久久乱码一区二区| 久久免费观看视频| 国产精品青草综合久久久久99| 韩日精品视频| 亚洲一区中文| 亚洲国产精品久久久久秋霞影院 | 99riav国产精品| 亚洲欧美日韩国产一区二区| 免费久久久一本精品久久区| 在线亚洲电影| 欧美理论电影网| 黄色成人小视频| 亚洲欧美国产77777| 亚洲国产你懂的| 久久久国产精品亚洲一区 | 国产精品久久久久久久久久直播| 国产一区二区三区精品久久久| 99精品国产福利在线观看免费 | 亚洲精品资源美女情侣酒店| 久久久久亚洲综合| 亚洲综合激情| 欧美午夜电影在线| 99精品欧美一区二区三区综合在线| 久久久成人网| 午夜视频在线观看一区二区| 欧美午夜片在线免费观看| 一区二区动漫| 亚洲区欧美区| 欧美激情在线有限公司| 亚洲国产欧美一区二区三区久久| 久久精品国产精品| 午夜精品久久久久久久99黑人| 欧美婷婷六月丁香综合色| 在线视频欧美精品| 亚洲理论在线| 欧美午夜不卡视频| 亚洲欧美另类国产| 亚洲影视中文字幕| 国产欧美一区二区视频| 久久精品国产亚洲高清剧情介绍 | 亚洲视频精选| 国产精品国产三级国产专播品爱网| 99re6这里只有精品| 日韩亚洲国产欧美| 欧美日韩在线不卡| 亚洲综合不卡| 亚洲欧美欧美一区二区三区| 国产欧美日本| 另类图片综合电影| 欧美激情一区二区三区在线视频观看 | 欧美国产精品人人做人人爱| 性欧美精品高清| 国内自拍视频一区二区三区 | 亚洲国产影院| 欧美喷水视频| 性欧美xxxx视频在线观看| 亚洲欧美一区在线| 极品日韩av| 亚洲黄网站在线观看| 欧美私人网站| 久久久精品五月天| 麻豆久久婷婷| 亚洲自拍另类| 久久久精品动漫| 日韩一区二区免费看| 亚洲一区二区三区在线播放| 激情久久中文字幕| 日韩亚洲视频在线| 国内免费精品永久在线视频| 亚洲国产91色在线| 国产精品毛片一区二区三区| 免费成人性网站| 欧美日韩在线一区| 美女福利精品视频| 国产精品国产三级国产aⅴ9色| 久久噜噜噜精品国产亚洲综合| 欧美国产一区在线| 久久久久久久97| 欧美日韩亚洲国产一区| 久久久国产精品一区| 欧美日韩亚洲不卡| 欧美激情一区二区三区在线视频| 国产精品久线观看视频| 欧美黄色成人网| 国产欧美一区二区在线观看| 亚洲欧洲在线一区| 亚洲缚视频在线观看| 亚洲欧美大片| 亚洲一区二区精品在线| 免费试看一区| 久久av一区二区三区| 欧美视频中文字幕| 亚洲欧洲久久| 亚洲丰满少妇videoshd| 性欧美大战久久久久久久久| 亚洲一区二区三区成人在线视频精品 | 亚洲视频在线二区| 久久久综合网站| 欧美在线观看视频一区二区| 欧美日韩一区不卡| 亚洲国产欧美日韩| 亚洲黄色在线| 久久综合综合久久综合| 老司机成人在线视频| 国产日韩欧美a| 午夜精品在线观看| 欧美在线999| 国产农村妇女毛片精品久久莱园子| 亚洲精品欧美极品| 在线视频一区二区| 欧美日韩黄视频| 亚洲免费观看视频| 久久综合给合| 亚洲性线免费观看视频成熟| 久久国产精品久久国产精品| 亚洲综合三区| 欧美日韩中文字幕日韩欧美| 91久久精品国产| 亚洲国产日韩欧美在线图片| 久久久www成人免费毛片麻豆| 久久女同精品一区二区| 精品88久久久久88久久久| 久久激情婷婷| 免费看的黄色欧美网站| 亚洲欧洲日本专区| 免费一区视频| 亚洲另类春色国产| 午夜在线观看欧美| 国产一区二区三区免费在线观看| 欧美亚洲午夜视频在线观看| 老司机免费视频一区二区三区| 亚洲观看高清完整版在线观看| 久久久人成影片一区二区三区 | 国产主播在线一区| 快播亚洲色图| 日韩亚洲一区二区| 久久疯狂做爰流白浆xx| 一区二区三区在线高清| 欧美激情亚洲精品| 亚洲女同精品视频| 免费日本视频一区| 亚洲色图在线视频| 狠狠88综合久久久久综合网| 欧美国产欧美亚洲国产日韩mv天天看完整 | 欧美日韩亚洲一区在线观看| 亚洲淫片在线视频| 免费亚洲网站| 亚洲无线一线二线三线区别av| 国产日韩精品在线| 欧美成人午夜77777| 亚洲一区三区电影在线观看| 欧美国产日韩免费| 午夜影院日韩| 亚洲人成网站999久久久综合| 欧美性做爰猛烈叫床潮| 久久久精品一区| 亚洲一区二区日本| 最新成人av在线| 久久一日本道色综合久久| 一区二区成人精品| 亚洲国产成人久久| 国产视频观看一区| 欧美日韩亚洲综合| 免费不卡在线视频| 欧美专区中文字幕| 亚洲图片欧美午夜| 欧美精品粉嫩高潮一区二区| 欧美电影免费观看| 久久精品一级爱片| 亚洲一区二区三区色| 91久久精品视频| 激情综合色综合久久| 国产一级揄自揄精品视频|