• <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>

            S.l.e!ep.¢%

            像打了激速一樣,以四倍的速度運轉,開心的工作
            簡單、開放、平等的公司文化;尊重個性、自由與個人價值;
            posts - 1098, comments - 335, trackbacks - 0, articles - 1
              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理
            IOCP的一個簡單封裝類(zz)
            #pragma once
            ?
            #include <winsock2.h>
            #pragma comment( lib, "ws2_32.lib" )
            ?
            const int OP_READ = 0;
            const int OP_WRITE = 1;
            const int OP_ACCEPT = 2;
            ?
            /*
            ???? OVERLAPPEDPLUS 結構體設計思路
            ???? OVERLAPPED 是一個固定的用于處理網絡消息事件返回值的結構體變量
            ???? 在完成端口和重疊I/O模型里用于返回消息事件的結果
            ???? 因為在處理網絡消息的時候,發送的是一個返回值的結構體指針,只要結構體
            ???? 的前面部分滿足系統的要求,在系統操作成功的時候也就會把這個結構體指針
            ???? 發回給用戶,我們只要在系統定義的結構體后面擴展一些自己的東西,就可以
            ???? 很輕松的確定該消息是誰發過來的。
            ???? 不過好像完成端口在設計的時候也滿足了這樣的需求,所以在這里我只是放入
            ???? 一些與系統連接有關的數據,用戶需要存放的數據這里就不在存放
            ???? 這里存儲與系統相關的數據有:
            ???? socket
            ???? OpCode 本次消息的操作類型(在完成端口的操作里面,是以消息通知系統,
            ???????? 讀數據/寫數據,都是要發這樣的消息結構體過去的,所以如果系統要同時
            ???????? 進行讀寫操作的話,就需要有一個變量來區分操作了)
            ?
            ???? WSABUF?? wbuf;????????????????? //?? 讀寫緩沖區結構體變量
            ???? DWORD??? dwBytes, dwFlags;?//?? 一些在讀寫時用到的標志性變量
            ???? char buf[4096];????????????????? //?? 自己的緩沖區
            ???? 上面的4個變量存放的是一些與消息相關的數據,都是一些操作上用到的,
            ???? 這些東西都是固定的,具體作用需要參考一下完成端口相關函數的參數接口
            */
            struct OVERLAPPEDPLUS
            {
            ???? OVERLAPPED??? ol;
            ???? SOCKET??????? s;
            ???? int OpCode;
            ???? WSABUF?? wbuf;
            ???? DWORD??? dwBytes, dwFlags;
            ???? char buf[4096];
            };
            ?
            class CIOCP
            {
            protected :
            ???? HANDLE g_hwThread;???? //?? 工作線程句柄
            ???? DWORD?m_wthreadID;
            ???? HANDLE g_haThread;???? //?? 連接線程句柄
            ???? DWORD?m_athreadID;
            public :
            ???? bool m_workThread;
            ???? bool m_acceptThread;
            ???? HANDLE m_hIocp;???????????? //?? 完成端口的句柄
            ???? SOCKET m_sSocket;
            ????
            public :
            ???? CIOCP(void);
            ???? ~CIOCP(void);
            ???? virtualvoid OnRead(void * p, char *buf, int len){};
            ???? virtualvoid OnAccept(SOCKET socket);
            ???? virtualvoid OnClose(void * p){};
            ???? bool SetIoCompletionPort(SOCKET socket, void *p, char *buf = NULL, int len = 0);
            ???????? //?? 把一個socket與一個自定義的結構體關聯到完成端口(相當于把socket與一個結構體變量進行綁定),
            ???????? //?? 這樣當發送上面3種網絡事件的時候,該結構體變量會再傳回給程序
            ???????? //?? 這樣就可以區分當前網絡事件是那個socket發出的
            ???? bool Init(void);
            ???? bool Listen(int port);
            ???? static DWORD __stdcall WorkThread(LPVOID Param);
            ???? static DWORD __stdcall AcceptThread(LPVOID Param);
            };
            ?
            ?
            ?
            ?
            ?
            ?
            class CIOCPClient: public CIOCP
            {
            protected :
            ???? SOCKET m_socket;
            public :
            ???? bool Connect(char *ip, int port);
            ???? void Send(char *buf, int len);
            };
            ?
            ?
            ?
            ?
            ?
            ?
            //////////////////////////////////////////////////////////////////////////////////////////
            //?? Iocp 實現文件
            ?
            #include "StdAfx.h"
            #include "iocp.h"
            ?
            static bool bInit = false;
            ?
            DWORD __stdcall CIOCP::WorkThread(LPVOID Param)
            {
            ???? CIOCP * pthis = (CIOCP *)Param;
            ?
            ???? void * re;
            ???? OVERLAPPED * pOverlap;
            ???? DWORD berByte;
            ???? while(pthis->m_workThread)
            ???? {
            ???????? int ret;
            ???????? ret = GetQueuedCompletionStatus(pthis->m_hIocp, &berByte, (LPDWORD)&re, (LPOVERLAPPED *)&pOverlap, INFINITE);
            ?
            ???????? if (ret == ERROR_SUCCESS)
            ???????? {
            ?
            ???????? }
            ?
            ???????? if (berByte == 0)
            ???????? {
            ????????????? //?? 客戶端斷開連接
            ????????????? pthis->OnClose(re);
            ????????????? OVERLAPPEDPLUS *olp = (OVERLAPPEDPLUS *)pOverlap;
            ????????????? closesocket(olp->s);
            ????????????? delete olp;??????? //?? 釋放 與socket綁定的結構體變量
            ????????????? continue;
            ???????? }
            ?
            ???????? if (re == NULL) return 0;
            ?
            ???????? OVERLAPPEDPLUS *olp = (OVERLAPPEDPLUS *)pOverlap;
            ?
            ???????? switch(olp->OpCode)
            ???????? {
            ???????? case OP_READ:
            ????????????? pthis->OnRead(re, olp->wbuf.buf, berByte);???? //?? 調用 OnRead() 通知應用程序,服務器收到來自客戶端的網絡數據
            ????????????? WSARecv(olp->s, &olp->wbuf, 1, &olp->dwBytes, &olp->dwFlags, &olp->ol, NULL);?//?? 繼續調用一個接收的 I/O 異步請求
            ????????????? break;
            ???????? default:
            ????????????? break;
            ???????? }
            ???? }
            ???? return 0;
            }
            ?
            DWORD __stdcall CIOCP::AcceptThread(LPVOID Param)
            {
            ???? CIOCP * pthis = (CIOCP *)Param;
            ???? while(pthis->m_acceptThread)
            ???? {
            ???????? SOCKET client;
            ???????? if ((client= accept(pthis->m_sSocket, NULL, NULL)) == INVALID_SOCKET)
            ???????? {
            ????????????? //?? 錯誤處理
            ???????? }
            ???????? pthis->OnAccept(client);??? //?? 調用 OnAccept()通知應用程序有新客戶端連接
            ????????
            ???? }
            ???? return 1;
            }
            ?
            CIOCP::CIOCP(void)
            {
            }
            ?
            CIOCP::~CIOCP(void)
            {
            }
            ?
            bool CIOCP::Init(void)
            {
            ???? if (bInit)
            ???????? returntrue;
            ?
            ???? WSADATA wsd;
            ???? if (WSAStartup(MAKEWORD(2,2), &wsd) != 0)
            ???????? returnfalse;
            ?
            ???? bInit = true;
            ???? returntrue;
            }
            ?
            bool CIOCP::Listen(int port)
            {
            ???? if (!bInit)
            ???????? if (!Init())
            ????????????? returnfalse;
            ?
            ???? m_sSocket = socket(AF_INET, SOCK_STREAM, 0);
            ?
            ???? if (m_sSocket == INVALID_SOCKET)
            ???????? returnfalse;
            ?
            ???? //SOCKADDR_IN addr;
            ???? sockaddr_in addr;
            ???? addr.sin_family = AF_INET;
            ???? addr.sin_port = htons(port);
            ???? addr.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
            ???? //addr.sin_addr.S_un.S_addr = inet_addr(ip);
            ?
            ???? if (bind(m_sSocket, (struct sockaddr *)&addr, sizeof(addr)) == SOCKET_ERROR)
            ???????? returnfalse;
            ?
            ???? if (listen(m_sSocket, 10) == SOCKET_ERROR)
            ???????? returnfalse;
            ?
            ???? if ((m_hIocp = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, NULL, 0)) == NULL)???? //?? 創建完成端口的句柄
            ???????? returnfalse;
            ?
            ???? this->m_acceptThread = true;
            ???? g_haThread = CreateThread(NULL, 0, AcceptThread, (LPVOID)this, 0, &m_athreadID);??? //?? 創建連接線程,用來接收客戶端的連接
            ?
            ???? this->m_workThread = true;
            ???? g_hwThread = CreateThread(NULL, 0, WorkThread, (LPVOID)this, 0, &m_wthreadID); //?? 創建工作線程,用來處理完成端口消息的
            ???? returntrue;
            }
            ?
            bool CIOCP::SetIoCompletionPort(SOCKET socket, void *p, char *buf, int len)
            {
            ???? if (CreateIoCompletionPort((HANDLE)socket, m_hIocp, (ULONG_PTR)p, 0) == NULL)
            ???????? returnfalse;
            ?
            ???? OVERLAPPEDPLUS *olp = new OVERLAPPEDPLUS;
            ???? memset(olp, 0, sizeof(OVERLAPPEDPLUS));
            ???? olp->s = socket;
            ???? if (buf)
            ???? {
            ???????? //?? 這里可以使用用戶自定義的緩沖區地址,如果用戶不想設置,也可以采用默認分配的緩沖區
            ???????? olp->wbuf.buf = buf;
            ???????? olp->wbuf.len = len;
            ???? }
            ???? else
            ???? {
            ???????? olp->wbuf.buf = olp->buf;
            ???????? olp->wbuf.len = 4096;
            ???? }
            ???? olp->OpCode = OP_READ;
            ???? int ret = WSARecv(olp->s, &olp->wbuf, 1, &olp->dwBytes, &olp->dwFlags, &olp->ol, NULL);
            ???? if (ret == SOCKET_ERROR)
            ???????? if (WSAGetLastError() != ERROR_IO_PENDING)
            ????????????? returnfalse;
            ???? returntrue;
            }
            ?
            void CIOCP::OnAccept(SOCKET socket)
            {
            ???? this->SetIoCompletionPort(socket, NULL);
            }
            ?
            //===================================================================================
            ?
            ?
            ?
            ?
            ?
            ?
            bool CIOCPClient::Connect(char *ip, int port)
            {
            ???????? //?? 連接服務器
            ???? if (!bInit)
            ???????? if (!Init())
            ????????????? returnfalse;
            ?
            ???? //?? 初始化連接socket
            ???? m_socket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
            ???? if (m_socket == SOCKET_ERROR)
            ???? {
            //?????? printf("cocket Create fail");
            ???????? returnfalse;
            ???? }
            ?
            ???? // 填寫服務器地址信息
            ???? // 端口為1982
            ???? // IP 地址為INADDR_ANY,注意使用htonl將IP地址轉換為網絡格式ServerAddr.sin_family = AF_INET;
            ???? sockaddr_in ClientAddr;
            ???? ClientAddr.sin_family = AF_INET;
            ???? ClientAddr.sin_port = htons(port);???
            ???? ClientAddr.sin_addr.s_addr = inet_addr(ip);
            ?
            ???? // 綁定監聽端口
            ???? bind(m_socket, (SOCKADDR *)&ClientAddr, sizeof(ClientAddr));
            ?
            ???? if (connect(m_socket, (SOCKADDR *)&ClientAddr, sizeof(ClientAddr)) == SOCKET_ERROR)
            ???? {
            ???????? returnfalse;
            ???? }
            ????
            ???? if ((m_hIocp = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, NULL, 0)) == NULL)???? //?? 創建完成端口的句柄
            ???????? returnfalse;
            ?
            ???? this->m_workThread = true;
            ???? g_hwThread = CreateThread(NULL, 0, WorkThread, (LPVOID)this, 0, &m_wthreadID); //?? 創建工作線程,用來處理完成端口消息的
            ?
            ???? this->SetIoCompletionPort(m_socket, &m_socket);??? //?? 設置完成端口監聽的socket
            ???? returntrue;
            }
            ?
            void CIOCPClient::Send(char *buf, int len)
            {
            ???? send(m_socket, buf, len, 0);
            }
            ?
            ?
            ?
            ?
            ?
            ?
            ?
            ?
            ?
            ///////////////////////////////////////////////////////////////////////////////////
            // IOCPclient 應用代碼
            ?
            #include "stdafx.h"
            #include "IOCP.h"
            #include "TClientSocket.h"
            class Iocp :public CIOCPClient
            {
            ???? ?void OnRead(void * p, char *buf, int len)
            ???? ?{
            ???????? ?printf(buf);
            ???????? ?Sleep(1000);
            ???????? ?this->Send(buf, len);
            文件:IOCP的簡單類封裝_源文件.rar
            大小:3KB
            下載:下載
            文件: IOCP的簡單應用實例.rar
            大小: 318KB
            下載: 下載
            ???? ?}
            };
            ?
            int _tmain(int argc, _TCHAR* argv[])
            {
            ???? Iocp iocp;
            ???? iocp.Init();
            ???? iocp.Connect("127.0.0.1", 4311);
            ???? iocp.Send("test\0", 5);
            ????
            ???? gets(newchar[1000]);
            ???? return 0;
            }

            發表于: 2007-03-26,修改于: 2007-03-26 16:27,已瀏覽959次,有評論0條 推薦 投訴

            Feedback

            # re: IOCP的一個簡單封裝類(zz) [轉]  回復  更多評論   

            2014-07-05 11:11 by dsa
            例子不能運行
            久久久久久久国产免费看| 国产巨作麻豆欧美亚洲综合久久| 久久国产欧美日韩精品免费| 久久久中文字幕日本| 蜜臀av性久久久久蜜臀aⅴ麻豆 | 国产精品久久久久9999| 亚洲伊人久久综合中文成人网| 狠狠色噜噜狠狠狠狠狠色综合久久| 欧美性猛交xxxx免费看久久久| 国产A级毛片久久久精品毛片| 日韩va亚洲va欧美va久久| 欧美噜噜久久久XXX| 久久丝袜精品中文字幕| 精品国产VA久久久久久久冰 | 国产精品99久久久久久人| 久久精品国产精品亚洲人人 | 77777亚洲午夜久久多喷| 久久一区二区三区99| 亚洲欧美日韩精品久久| 亚洲成人精品久久| 亚洲AV无码久久精品狠狠爱浪潮| A级毛片无码久久精品免费| 久久久久久久精品成人热色戒| 久久香综合精品久久伊人| 国产成人久久777777| 99久久国产热无码精品免费 | segui久久国产精品| 9久久9久久精品| 精品无码久久久久国产| 日韩精品久久久肉伦网站| 一本久久知道综合久久| 伊色综合久久之综合久久| 久久男人AV资源网站| 久久久久国产视频电影| 久久精品亚洲欧美日韩久久| 91精品国产91久久| 99久久国产主播综合精品| 国产成人无码精品久久久免费| 日本久久久久久中文字幕| 久久午夜免费视频| 国产欧美久久久精品影院|