• [C++ 网络协议] IOCP(Input Output Completion Port)


    1.什么是IOCP

    IOCP(Input Output Completion Port)输入输出完成端口。其实就是基于重叠I/O的一种改进的模型。

    重叠I/O具有缺点:重复调用非阻塞模式的accpet函数和以进入alertablewait状态为目的的SleepEx函数会影响程序性能

    而IOCP提供的解决方案便是:让主线程调用accept函数,单独创建至少一个线程来负责所有I/O的前后处理

    但请不要过分关注在线程上,主要还是如下问题:

            1.I/O是否以非阻塞模式工作?

            2.如何确定非阻塞模式的I/O是否完成?

    2.分阶段实现IOCP程序

    2.1 实现原理

    IOCP会将已完成的I/O信息注册到CP对象(Completion Port完成端口),而我们就可以通过CP对象来获取I/O是否完成的信息,所以有下面两项工作:

    • 创建完成端口对象
    • 建立完成端口对象和套接字之间的联系 

    此时的套接字必须赋予重叠属性。

    2.2 创建CP对象

    1. #include
    2. HANDLE CreateIoCompletionPort(
    3. HANDLE fileHandle, //创建CP对象时传递INVALID_HANDLE_VALUE
    4. HANDLE ExistingCompletionPort, //创建CP对象时传递NULL
    5. ULONG_PTR CompletionKey, //创建CP对象时传递0
    6. DWORD NumberOfConcurrentThreads //分配给CP对象的用于处理I/O的线程数。
    7. //例如:该参数为2时,说明分配给CP对象的可以同时运行的线程数最多为2个
    8. //如果为0时,那么系统中CPU的个数就是可同时运行的最大线程数
    9. );
    10. 成功返回CP对象句柄
    11. 失败返回NULL

    2.3 创建和套接字连接完成的端口对象

    1. #include
    2. HANDLE CreateIoCompletionPort(
    3. HANDLE FileHandle, //要连接到CP对象的套接字句柄
    4. HANDLE ExistingCompletionPort, //要连接套接字的CP对象句柄
    5. ULONG_PTR CompletionKey, //传递已完成I/O相关信息
    6. DWORD NumberOfConcurrentThreads //无论传递何值,只要第二个参数非NULL就会被忽略
    7. );
    8. 成功返回CP对象句柄
    9. 失败返回NULL

    函数功能:将FileHandle句柄指向的套接字和ExistingCompletionPort指向的CP对象相连。

    调用此函数后:只要针对FileHandle的I/O完成,相关信息就会注册到ExistingCompletionPort里。

    注意:第三个参数“传递已完成I/O相关信息”的意思是,你可以像重叠I/O里使用Complition routine来确认I/O方式里把相关信息填写到hEvent里的那样,写入其他信息,这样当I/O完成就可以获取了。

    2.4 确认完成端口已完成的I/O和线程I/O处理

    1. #include
    2. BOOL GetQueuedCompletionStatus(
    3. HANDLE CompletionPort, //注册有已完成I/O信息的CP对象句柄
    4. LPDWORD lpNumberOfBytes, //保存I/O过程中传输的数据大小的变量地址值
    5. PULONG_PTR lpCompletionKey, //保存CreateIoCompleytionPort函数第三个参数值得变量地址值
    6. LPOVERLAPPED* lpOverlapped, //保存调用WSASend、WSARecv函数时传递的OVERLAPPED结构体地址的变量地址值
    7. DWORD dwMilliseconds //超时信息,超过该指定时间后将返回FALSE并跳出函数。
    8. //传递INFINITE时,程序将阻塞,直到已完成I/O信息写入CP对象
    9. );
    10. 成功返回TRUE
    11. 失败返回FALSE

    注意:

    • 调用此函数的线程数量不能超过CreateIoCompletionPort时指定的线程数。
    • 此函数并不知道当前是输入信息状态还是输出信息状态,需要自行判断。

    3. 实现IOCP模型的回声服务器端

    思路:每连接一个客户端就创建一个线程,然后主线程里先接收一次数据,在子线程里通过GetQueuedCompletionStatus函数阻塞住线程,判断I/O状态,接着把接收的数据发送给客户端,再次进入接收状态,如此循环通信。

    变量:

    struct ClientInfo结构体:存有套接字和套接字地址族信息,在CreateIoCompletionPort函数里,建立套接字和CP的连接的时候,当做第三参数传入

    struct CPInfo结构体:存有一个OVERLAPPED、WSABUF信息,以及还有一个int型用来判断当前是RECV还是SEND,在执行WSARecv函数时当做第六个参数进行传入。运用下面的知识点,所以可以在子线程执行GetQueuedCompletionStatus函数时,取得的第一个成员的地址,也就是这整个结构体的地址。

    知识点:结构体变量地址值与结构体第一个成员的地址值相同。

    1. struct CPInfo
    2. {
    3. OVERLAPPED overlapped;
    4. WSABUF wsabuf;
    5. int mode; //0:RECV 1:SEND
    6. };
    7. CPInfo data;
    8. if(&data==&data.overlapped)
    9. {
    10. std::cout<<"TRUE"<
    11. }
    12. else
    13. {
    14. std::cout<<"FALSE"<
    15. }
    16. 输出TRUE
    1. #define _WINSOCK_DEPRECATED_NO_WARNINGS
    2. #include
    3. #include
    4. #include
    5. #include
    6. #include
    7. #include
    8. struct ClientInfo
    9. {
    10. SOCKET socket;
    11. sockaddr_in socketAddr;
    12. };
    13. struct CPInfo
    14. {
    15. OVERLAPPED overlapped;
    16. WSABUF wsabuf;
    17. int mode; //0:RECV 1:SEND
    18. };
    19. unsigned WINAPI threadClient(void* arg);
    20. int main()
    21. {
    22. WSADATA wsaData;
    23. if (0 != WSAStartup(MAKEWORD(2, 2), &wsaData))
    24. {
    25. std::cout << "start up fail!" << std::endl;
    26. return 0;
    27. }
    28. SOCKET server = WSASocket(PF_INET, SOCK_STREAM, IPPROTO_TCP, NULL, 0, WSA_FLAG_OVERLAPPED);
    29. if (server == INVALID_SOCKET)
    30. {
    31. std::cout << "socket fail!" << std::endl;
    32. return 0;
    33. }
    34. int mode = 1;
    35. ioctlsocket(server, FIONBIO, (u_long*)&mode);
    36. sockaddr_in serverAddr;
    37. memset(&serverAddr, 0, sizeof(serverAddr));
    38. serverAddr.sin_family = AF_INET;
    39. serverAddr.sin_addr.s_addr = htonl(INADDR_ANY);
    40. serverAddr.sin_port = htons(9130);
    41. if (SOCKET_ERROR == bind(server, (sockaddr*)&serverAddr, sizeof(serverAddr)))
    42. {
    43. std::cout << "bind fail!" << std::endl;
    44. return 0;
    45. }
    46. if (SOCKET_ERROR == listen(server, 2))
    47. {
    48. std::cout << "listen fail!" << std::endl;
    49. return 0;
    50. }
    51. while (true)
    52. {
    53. sockaddr_in clientAddr;
    54. memset(&clientAddr, 0, sizeof(clientAddr));
    55. int clientAddrLen = sizeof(clientAddr);
    56. SOCKET client = accept(server, (sockaddr*)&clientAddr, &clientAddrLen);
    57. if (client == SOCKET_ERROR)
    58. {
    59. if (WSAGetLastError() == WSAEWOULDBLOCK) //说明此时没有客户端连接
    60. {
    61. continue;
    62. }
    63. std::cout << "accept fail!" << std::endl;
    64. }
    65. else
    66. {
    67. HANDLE cpObject = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, 0);
    68. if (cpObject == NULL)
    69. {
    70. std::cout << "Create CP fail!" << std::endl;
    71. continue;
    72. }
    73. ClientInfo* clientinfo = new ClientInfo();
    74. clientinfo->socket = client;
    75. clientinfo->socketAddr = clientAddr;
    76. CreateIoCompletionPort((HANDLE)client, cpObject, (ULONG_PTR)clientinfo, 0);
    77. unsigned threadId;
    78. if (0 == _beginthreadex(NULL, 0, threadClient, (void*)&cpObject, 0, &threadId)) //创建一个线程
    79. {
    80. std::cout << "thread create fail!" << std::endl;
    81. continue;
    82. }
    83. CPInfo* cpinfo = new CPInfo();
    84. cpinfo->mode = 0;
    85. memset(&cpinfo->overlapped, 0, sizeof(cpinfo->overlapped));
    86. char buff[1024];
    87. cpinfo->wsabuf.buf = buff;
    88. cpinfo->wsabuf.len = sizeof(buff);
    89. DWORD readLen;
    90. DWORD flag = 0;
    91. WSARecv(client, &cpinfo->wsabuf, 1, &readLen, &flag, &cpinfo->overlapped, NULL);
    92. }
    93. }
    94. closesocket(server);
    95. WSACleanup();
    96. }
    97. unsigned WINAPI threadClient(void* arg)
    98. {
    99. HANDLE cpObject = *(HANDLE*)arg;
    100. CPInfo* cpinfo;
    101. ClientInfo* clientinfo;
    102. while (true)
    103. {
    104. DWORD readLen;
    105. GetQueuedCompletionStatus(cpObject, &readLen, (PULONG_PTR)&clientinfo, (LPOVERLAPPED*)&cpinfo, INFINITE);
    106. if (readLen == 0)
    107. {
    108. std::cout << "客户端:" << inet_ntoa(clientinfo->socketAddr.sin_addr) << "断开连接!" << std::endl;
    109. break;
    110. }
    111. if (cpinfo->mode == 0) //recv
    112. {
    113. std::cout << "客户端发来的消息:" << cpinfo->wsabuf.buf << std::endl;
    114. DWORD flag = 0;
    115. cpinfo->mode = 1;
    116. WSASend(clientinfo->socket, &cpinfo->wsabuf, 1, &readLen, flag, &cpinfo->overlapped, NULL);
    117. CPInfo* cpinfo2 = new CPInfo();
    118. cpinfo2->mode = 0;
    119. memset(&cpinfo2->overlapped, 0, sizeof(cpinfo2->overlapped));
    120. char buff[1024];
    121. cpinfo2->wsabuf.buf = buff;
    122. cpinfo2->wsabuf.len = sizeof(buff);
    123. DWORD readLen2;
    124. WSARecv(clientinfo->socket, &cpinfo2->wsabuf, 1, &readLen2, &flag, &cpinfo2->overlapped, NULL);
    125. }
    126. else //send
    127. {
    128. delete cpinfo;
    129. }
    130. }
    131. CloseHandle(cpObject);
    132. closesocket(clientinfo->socket);
    133. return 0;
    134. }
  • 相关阅读:
    Toronto Research Chemicals BTK抑制剂丨ACP-5197
    广州华锐互动:VR互动实训内容编辑器助力教育创新升级
    开源语言大模型的正确姿势
    redis内存描述
    ERROR: [Synth 8-439] module ‘xxx‘ not found not found 错误解决办法
    全光网络技术、标准、应用现状及展望
    【Oracle】Oracle系列之九--Oracle常用函数
    网站安全-行为式验证码
    有了低代码,二次开发都不是事!
    2022 Gartner RPA魔力象限发布,两家国产厂商入选,超自自动化成重点
  • 原文地址:https://blog.csdn.net/A_ns_wer_/article/details/133387387