1. select模型

select模型主要借助于apiselect来实现,所以先介绍一下select函数

  1. int select(
  2. int nfds, // 忽略,仅是为了与 Berkeley 套接字兼容
  3. fd_set* readfds, // 指向一个套接字集合,用来检查其可读性
  4. fd_set* writefds, // 指向一个套接字集合,用来检查其可写性
  5. fd_set* exceptfds, // 指向一个套接字集合,用来检查错误
  6. const struct timeval* timeout // 指定此函数等待的最长时间,如果为 NULL,则最长时间为无限大
  7. );
  8. // fd_set结构体
  9. typedef struct fd_set {
  10. u_int fd_count; // 下面数组的大小
  11. SOCKET fd_array[FD_SETSIZE]; // 套接字句柄数组
  12. } fd_set;
  13. FD_ZERO(*set) //初始化 set 为空集合。集合在使用前应该总是清空
  14. FD_CLR(s, *set) //从 set 移除套接字 s
  15. FD_ISSET(s, *set) //检查 s 是不是 set 的成员,如果是返回 TRUE
  16. FD_SET(s, *set) //添加套接字到集合

2.例子

一个简单的流程:

  1. 初始化套接字集合 fdSocket,向这个集合添加监听套接字句柄。
  2. 将 fdSocket 集合的拷贝 fdRead 传递给 select 函数,当有事件发生时, select 函数移除 fdRead 集合中没有未决 I/O 操作的套接字句柄,然后返回。
  3. 比较原来 fdSocket 集合与 select 处理过的 fdRead 集合,确定哪些套接字有未决 I/O,并进一步处理这些 I/O。
  4. 回到第 2 步继续进行选择处理。
  1. CInitSock theSock; // 初始化 Winsock 库
  2. int main()
  3. {
  4. USHORT nPort = 4567; // 此服务器监听的端口号
  5. // 创建监听套接字
  6. SOCKET sListen = ::socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
  7. sockaddr_in sin;
  8. sin.sin_family = AF_INET;
  9. sin.sin_port = htons(nPort);
  10. sin.sin_addr.S_un.S_addr = INADDR_ANY;
  11. // 绑定套接字到本地机器
  12. if (::bind(sListen, (sockaddr*)&sin, sizeof(sin)) == SOCKET_ERROR)
  13. {
  14. printf(" Failed bind() \n");
  15. return -1;
  16. }
  17. // 进入监听模式
  18. ::listen(sListen, 5);
  19. // select 模型处理过程
  20. // 1)初始化一个套接字集合 fdSocket,添加监听套接字句柄到这个集合
  21. fd_set fdSocket; // 所有可用套接字集合
  22. FD_ZERO(&fdSocket);
  23. FD_SET(sListen, &fdSocket);
  24. while (TRUE)
  25. { // 2)将 fdSocket 集合的一个拷贝 fdRead 传递给 select 函数,
  26. // 当有事件发生时, select 函数移除 fdRead 集合中没有未决 I/O 操作的套接字句柄,然后返回。
  27. fd_set fdRead = fdSocket;
  28. int nRet = ::select(0, &fdRead, NULL, NULL, NULL);
  29. if (nRet > 0)
  30. { // 3)通过将原来 fdSocket 集合与 select 处理过的 fdRead 集合比较,
  31. // 确定都有哪些套接字有未决 I/O,并进一步处理这些 I/O。
  32. for (int i = 0; i < (int)fdSocket.fd_count; i++)
  33. {
  34. if (FD_ISSET(fdSocket.fd_array[i], &fdRead))
  35. {
  36. if (fdSocket.fd_array[i] == sListen) // ( 1)监听套接字接收到新连接
  37. {
  38. if (fdSocket.fd_count < FD_SETSIZE)
  39. {
  40. sockaddr_in addrRemote;
  41. int nAddrLen = sizeof(addrRemote);
  42. SOCKET sNew =
  43. ::accept(sListen, (SOCKADDR*)&addrRemote, &nAddrLen);
  44. FD_SET(sNew, &fdSocket);
  45. printf("接收到连接( %s) \n", ::inet_ntoa(addrRemote.sin_addr));
  46. }
  47. else
  48. {
  49. printf(" Too much connections! \n");
  50. continue;
  51. }
  52. }
  53. else
  54. {
  55. char szText[256];
  56. int nRecv = ::recv(fdSocket.fd_array[i], szText, strlen(szText), 0);
  57. if (nRecv > 0) // ( 2)可读
  58. {
  59. szText[nRecv] = '\0';
  60. printf("接收到数据: %s \n", szText);
  61. }
  62. else // ( 3)连接关闭、重启或者中断
  63. {
  64. ::closesocket(fdSocket.fd_array[i]);
  65. FD_CLR(fdSocket.fd_array[i], &fdSocket);
  66. }
  67. }
  68. }
  69. }
  70. }
  71. else
  72. {
  73. printf(" Failed select() \n");
  74. break;
  75. }
  76. }
  77. return 0;
  78. }

3.好处

使用 select 的好处是程序能够在单个线程内同时处理多个套接字连接,这避免了阻塞模

式下的线程膨胀问题。但是,添加到 fd_set 结构的套接字数量是有限制的,默认情况下,最

大值是 FD_SETSIZE,它在 winsock2.h 文件中定义为 64。为了增加套接字数量,应用程序可

以将 FD_SETSIZE 定义为更大的值(这个定义必须在包含 winsock2.h 之前出现)。不过,自

定义的值也不能超过 Winsock 下层提供者的限制(通常是 1024)

Windsock套接字I/O模型学习 --- 第二章的更多相关文章

  1. Windsock套接字I/O模型学习 --- 第一章

    1. I/O模型共有以下几种: 阻塞(blocking)模型 选择(select)模型 WSAAsyncSelect模型 WSAEventSelect模型 重叠(overlapped)模型 完成端口( ...

  2. Windsock套接字I/O模型学习 --- 第三章

    1. WSAAsyncSelect 模型 WSAAsyncSelect 模型比较简单,是为了适应Windows的消息驱动环境而设置的,WSAAsyncSelect 函数自动把套接字设为非阻塞模式.MF ...

  3. 套接字I/O模型之WSAEventSelect

    今天我又学习了一种新的套接字I/O模型------WSAEventSelect,他与WSAAsyncSelect一样也是一种异步事件通知模型,不同的是WSAAsyncSelect是与窗口句柄关联在一起 ...

  4. 套接字I/O模型-select

    共有6种类型套接字I/O模型.blocking(阻塞),select(选择),WSAAsyncSelect(异步选择),WSAEventSelect(事件选择),overlapped(重叠),comp ...

  5. oracle学习 第二章 限制性查询和数据的排序 ——03

    这里.我们接着上一小节2.6留下的问题:假设要查询的字符串中含有"_"或"%".又该如何处理呢? 開始今天的学习. 2.7  怎样使用转义(escape)操作符 ...

  6. 套接字I/O模型-重叠I/O

    重叠模型的基本设计原理是让应用程序使用重叠的数据结构,一次投递一个或多个WinsockI/O请求.针对那些提交的请求,在它们完成之后,应用程序可为它们提供服务.模型的总体设计以Windows重叠I/O ...

  7. 套接字I/O模型-WSAAsyncSelect

    利用这个异步I/O模型,应用程序可在一个套接字上接收以Windows消息为基础的网络事件通知.WSAAsyncSelect和WSAEventSelect提供读写数据能力的异步通知,但它们不提供异步数据 ...

  8. 套接字I/O模型-完成端口IOCP

    “完成端口”模型是迄今为止最为复杂的一种I/O模型.然而,假若一个应用程序同时需要管理为数众多的套接字,那么采用这种模型,往往可以达到最佳的系统性能!但不幸的是,该模型只适用于Windows NT和W ...

  9. 套接字I/O模型-WSAEventSelect(转载)

    和WSAAsyncSelect类似,它也允许应用程序在一个或多个套接字上,接收以事件为基础的网络事件通知. 该模型最主要的区别是在于网络事件是由对象句柄完成的,而不是通过窗口例程完成. 事件通知 事件 ...

随机推荐

  1. bashrc - PS1(提示符配色)

    PS1设置: liBlack="\[\033[0;30m\]"boBlack="\[\033[1;30m\]"liRed="\[\033[0;31m\ ...

  2. 设计模式 装饰模式(Decorator)

    设计模式 装饰模式(Decorator) @author ixenos 装饰模式是什么 1.装饰模式以对客户端透明的方式对象的功能,是继承关系的一个替代方案,但装饰模式可以在不创造更多子类的情况下,对 ...

  3. MySQL数据类型和属性

    日期和时间数据类型 MySQL数据类型 含义 date 3字节,日期,格式:2014-09-18 time 3字节,时间,格式:08:42:30 datetime 8字节,日期时间,格式:2014-0 ...

  4. HttpPost与HttpVerbs.Post属性的区别

    1. the HttpPost attribute is a short for the HttpVerbs.Post one but since MVC 2.0. 从MVC 2.0之后HttpPos ...

  5. .Net Core 常见问题整理

    1.安装时报0x80070490 找不到元素 这里应该是vs只装了web没有装c++ 下载一个 VC_redist.x64.exe 安装就行了 https://github.com/dotnet/co ...

  6. Openjudge-计算概论(A)-年龄与疾病

    描述: 某医院想统计一下某项疾病的获得与否与年龄是否有关,需要对以前的诊断记录进行整理. 输入共2行,第一行为过往病人的数目n(0 < n <= 100),第二行为每个病人患病时的年龄.输 ...

  7. 修改textField的placeholder的字体和颜色

    textField.placeholder = @"username is in here!"; [textField setValue:[UIColor redColor] fo ...

  8. A - Space Elevator(动态规划专项)

    A - Space Elevator Time Limit:1000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64u ...

  9. 可写的计算监控(Writable computed observables)

    新手可忽略此小节,可写依赖监控属性真的是太advanced了,而且大部分情况下都用不到. 一般情况下,计算监控的值是通过其他监控属性的值计算出来的,因此它是只读的.这个看似很奇怪,我们有什么办法可以让 ...

  10. js iframe跨域访问

    1.什么是跨域? 2.前台解决跨域几种方法 2.1 动态创建script 2.2 使用document.domain 2.3使用HTML5新属性postMessage 2.4 利用iframe和loc ...