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

kenlistian

厚積薄發(fā). 勤為槳,思為帆

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

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

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

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

#define PORT 5150
#define DATA_BUFSIZE 8192

#pragma comment(lib, "Ws2_32")

typedef struct                        //這個玩意就是灌數(shù)據(jù),取數(shù)據(jù)的一個自定義數(shù)據(jù)結(jié)構(gòu)

                                              //和那個wm_data差不了多少,不過就是老要塞一個OverLapped結(jié)構(gòu),
{
   OVERLAPPED Overlapped;
   WSABUF DataBuf;
   CHAR Buffer[DATA_BUFSIZE];                    
   DWORD BytesSEND;                                 //發(fā)送字節(jié)數(shù)
   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次,這是第一次調(diào)用,至于為什么?我問問你
   //
   if ((CompletionPort = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, 0)) == NULL)
   {
      printf( "CreateIoCompletionPort failed with error: %d\n", GetLastError());
      return;
   }
   //老套子api,不談也罷
   GetSystemInfo(&SystemInfo);
  
   //發(fā)現(xiàn)2個CPU,那就開個雙倍的線程跑吧
   for(i = 0; i < SystemInfo.dwNumberOfProcessors * 2; i++)
   {
      HANDLE ThreadHandle;
     
      //
      //完成端口掛到線程上面來了,就像管子把灌數(shù)據(jù)的和讀數(shù)據(jù)的兩頭都連上了,           
     //
      if ((ThreadHandle = CreateThread(NULL, 0, ServerWorkerThread, CompletionPort,
         0, &ThreadID)) == NULL)
      {
         printf("CreateThread() failed with error %d\n", GetLastError());
         return;
      }     
      CloseHandle(ThreadHandle);
   }

   //
   //啟動一個監(jiān)聽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;
   }

   //
   // 監(jiān)聽端口打開,就開始在這里循環(huán),一有socket連上,WSAAccept就創(chuàng)建一個socket,
   // 這個socket 又和完成端口聯(lián)上,
   //
   // 嘿嘿,完成端口第二次調(diào)用那個createxxx函數(shù),為什么,留給人思考思考可能更深刻,
   // 反正這套路得來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連起來
     //就像你把漏斗接到管子口上,開始要灌數(shù)據(jù)了
     //
      if (CreateIoCompletionPort((HANDLE) Accept, CompletionPort, (DWORD) PerHandleData,
         0) == NULL)
      {
         printf("CreateIoCompletionPort failed with error %d\n", GetLastError());
         return;
      }
     
      //
      //清管子的數(shù)據(jù)結(jié)構(gòu),準備往里面灌數(shù)據(jù)
      //
      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接到了數(shù)據(jù),就放到PerIoData中,而perIoData又通過線程中的函數(shù)取出,
     //
      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;
         }
      }
   }
}

//
//線程一但調(diào)用,就老在里面循環(huán),
// 注意,傳入的可是完成端口啊,就是靠它去取出管子中的數(shù)據(jù)
//
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)
   {
      //
      //在這里檢查完成端口部分的數(shù)據(jù)buf區(qū),數(shù)據(jù)來了嗎?
      // 這個函數(shù)參數(shù)要看說明,
      // PerIoData 就是從管子流出來的數(shù)據(jù),
      //PerHandleData 也是從管子里取出的,是何時塞進來的,
     //就是在建立第2次createIocompletionPort時
    // 

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

      // 檢查數(shù)據(jù)傳送完了嗎
      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;
      }    
     //
    //看看管子里面有數(shù)據(jù)來了嗎?=0,那是剛收到數(shù)據(jù)
    //
      if (PerIoData->BytesRECV == 0)
      {
         PerIoData->BytesRECV = BytesTransferred;
         PerIoData->BytesSEND = 0;
      }
      else   //來了,
      {
         PerIoData->BytesSEND += BytesTransferred;
      }
  
      //
      // 數(shù)據(jù)沒發(fā)完?繼續(xù)send出去
     //
     if (PerIoData->BytesRECV > PerIoData->BytesSEND)
      {

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

       //1個字節(jié)一個字節(jié)發(fā)送發(fā)送數(shù)據(jù)出去
         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)  編輯 收藏 引用

只有注冊用戶登錄后才能發(fā)表評論。
網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲午夜一级| 亚洲日本中文字幕| 久久国产手机看片| 久久本道综合色狠狠五月| 亚洲自拍偷拍色片视频| 亚洲国产精品久久91精品| 中文在线资源观看网站视频免费不卡| 一区一区视频| 亚洲欧洲一区二区三区久久| 日韩一级视频免费观看在线| 亚洲午夜精品网| 久久精品国产精品| 欧美aⅴ99久久黑人专区| 欧美大尺度在线| 亚洲精品中文字幕在线| 亚洲欧美成aⅴ人在线观看| 国产精品99久久久久久久vr| 欧美一区综合| 欧美丰满高潮xxxx喷水动漫| 国产精品高潮久久| 久久天天躁狠狠躁夜夜av| 国产精品久久久久91| 欧美区国产区| 久久精品卡一| 亚洲免费视频中文字幕| 亚洲天堂网在线观看| 香蕉成人啪国产精品视频综合网| 国产日韩精品入口| 国产精品亚洲第一区在线暖暖韩国| 狠狠色噜噜狠狠色综合久| 99re亚洲国产精品| 欧美亚洲一区| 久久免费观看视频| 一区二区欧美在线观看| 欧美在线视频不卡| 欧美美女日韩| 国产亚洲欧美一区二区| 亚洲激情视频在线观看| 性欧美超级视频| 亚洲精品国产精品国自产在线| 欧美一区二区三区在线免费观看| 欧美激情按摩| 一区二区亚洲精品| 亚洲男人的天堂在线| 亚洲福利视频二区| 久久激情五月激情| 国产精品多人| 亚洲一区二区黄| 亚洲精品在线观| 久久久久久亚洲精品不卡4k岛国| 国产精品青草久久| 亚洲性视频h| 亚洲麻豆av| 欧美精品亚洲精品| 亚洲日本视频| 亚洲二区视频在线| 老司机一区二区| 国产视频不卡| 久久久久国色av免费观看性色| av成人毛片| 国产精品久久看| 欧美视频免费| 在线亚洲国产精品网站| 欧美成人亚洲成人日韩成人| 久久爱www| 国产免费亚洲高清| 欧美在线视频免费播放| 午夜精品电影| 国产老肥熟一区二区三区| 欧美一区二区性| 欧美一级理论性理论a| 国产精品女人久久久久久| 亚洲一区二区动漫| 亚洲欧美一区二区三区久久 | 欧美一二区视频| 亚洲午夜高清视频| 国产乱码精品一区二区三区五月婷| 午夜精品久久久久久久99水蜜桃| 一区二区三区.www| 国产欧美在线观看一区| 久久精品国产99| 久久国产精品色婷婷| 激情综合视频| 亚洲国产精品一区| 欧美女人交a| 欧美亚洲一级片| 久久国产黑丝| 亚洲电影在线看| 亚洲精品少妇30p| 国产精品色在线| 国产日韩精品入口| 欧美激情一区在线观看| 久久国产精品99国产精| 一区二区三区三区在线| 亚洲视频在线视频| 在线日韩视频| 国产亚洲欧美一级| 国产精品久久久久永久免费观看| 午夜精品久久久久| 影院欧美亚洲| 99re热这里只有精品免费视频| 国产精品乱码妇女bbbb| 久热精品视频在线免费观看 | 久久精品国产免费| 亚洲国产欧美国产综合一区| 亚洲婷婷综合色高清在线| 一区在线影院| 日韩午夜免费视频| 在线观看国产精品网站| 一区二区欧美激情| 狠色狠色综合久久| 在线视频亚洲欧美| 亚洲第一网站| 欧美一区二区三区四区视频 | 久久精品一区四区| 欧美日韩三级一区二区| 久久久美女艺术照精彩视频福利播放| 欧美精品18+| 亚洲精品视频一区| 9色国产精品| 亚洲美女免费精品视频在线观看| 99re热这里只有精品视频| 亚洲激情在线播放| 久久精品视频免费播放| 午夜在线成人av| 国产精品高潮呻吟久久av黑人| 亚洲国产三级网| 伊人色综合久久天天五月婷| 亚洲欧美综合v| 午夜亚洲伦理| 欧美午夜精品久久久久免费视 | 亚洲精品看片| 久久久久9999亚洲精品| 久久精品30| 国产欧美日韩高清| 亚洲一区久久久| 午夜精品视频在线| 国产精品麻豆成人av电影艾秋 | 99国内精品久久久久久久软件| 亚洲另类在线视频| 欧美激情一区二区| 亚洲三级色网| 一区二区av在线| 欧美理论大片| 日韩午夜在线电影| 日韩亚洲在线| 欧美视频二区| 最新日韩在线| 毛片一区二区三区| 亚洲第一黄色网| 亚洲精品中文字幕在线观看| 欧美国产精品人人做人人爱| 亚洲精品在线二区| 亚洲自啪免费| 国内一区二区三区| 久久亚洲二区| 亚洲黑丝一区二区| 亚洲九九爱视频| 欧美视频亚洲视频| 香蕉久久a毛片| 亚洲大片在线| 欧美一级日韩一级| 久久亚洲二区| 亚洲毛片播放| 午夜精品视频网站| 黄色一区三区| 欧美激情久久久久| 在线视频欧美日韩| 午夜激情一区| 亚洲国产高清在线观看视频| 欧美国产在线观看| 亚洲一区二区三区影院| 久久九九电影| 亚洲国产精品va在看黑人| 欧美精品七区| 亚洲一区二区三区中文字幕在线| 亚洲欧美激情视频| 亚洲第一免费播放区| 欧美系列精品| 亚洲欧洲av一区二区| 亚洲国产精品va在线看黑人| 亚洲免费中文字幕| 亚洲区国产区| 国产在线国偷精品产拍免费yy| 葵司免费一区二区三区四区五区| 亚洲免费av电影| 久久久国产一区二区三区| 亚洲精品免费一区二区三区| 国产日韩欧美成人| 欧美极品aⅴ影院| 欧美一区精品| 亚洲午夜精品国产| 亚洲精品乱码久久久久久久久| 欧美一区三区三区高中清蜜桃| 亚洲美女电影在线| 狠狠色噜噜狠狠狠狠色吗综合| 欧美无砖砖区免费| 欧美精品导航| 久久精品国产亚洲一区二区| 亚洲午夜精品久久|