一.Windows I/O模型之选择(select)模型

1.选择(select)模型:
选择模型:通过一个fd_set集合管理套接字,在满足套接字需求后,通知套接字。让套接字进行工作。避免套接字进入阻塞模式,进行无谓的等待。选择模型的核心的FD_SET集合和select函数。通过该函数,我们可以们判断套接字上是否存在数据,或者能否向一个套接字写入数据。

2.select函数:
int select(
    int nfds,//忽略,只是为了兼容而存在。
    fd_set FAR* readfds,//可读性检查(有数据可读入,连接关闭,重设,终止)
    fd_set FAR* writefds,//可写性检查(有数据可发出)
    fd+set FAR* exceptfds,//带外数据检查(带外数据)
    const struct timeval FAR* timeout//超时
    );

3.select模型的工作步骤:
(1)首先把套接字加入到fd_set集合
(2)检查套接字的可读写性
(3)检查套接字是否还在fd_set集合上
(4)处理数据

4.在三个参数中(readfds、writefds和exceptfds),任何两个都可以是空值;但是,至少有一个不能为空值!最后一个参数timeout对应的是一个指针,它指向一个timeval结构,用于决select最多等待I/O操作完成多久的时间。如timeout是一个空指针,那么select函数会无限期地“等待下去,直到至少有一个套接字符合指定的条件后返回。select成功完成后,会在fd_set集合中,返回未完成的I/O操作的套接字句柄的总量。若超时,便会返回0。不管由于什么原因,假如select调用失败,都会返回SOCKET_ERROR错误。

5.timeval结构体定义:
struct timeval
{
    long tv_sec;//秒数
    long tv_usec;//毫秒数
};

6.fd_set集合:用select函数对套接字进行监视之前,必须要将套接字分配给一个fd_set集合,设置好读、写以及带外数据的fd_set结构。将一个套接字分配给任何一个集合后,再来调用select进行监视,便可知道一个套接字上是否正在发生上述的I/O活动。Winsock提供了下列宏操
作,对fd_set进行处理和检查:
    FD_ZERO(*set):初始化set
    FD_SET(s, *set):将套接字s加入集合set
    FD_CLR(s, *set):从set中删除套接字s。
    FD_ISSET(s,*set):检查s是否还在集合set上,在调用select函数之前必须对此进行判断。

7.select模型的工作步骤:
(1) 使用FD_ZERO宏,初始化自己感兴趣的每一个fd_set。
(2) 使用FD_SET宏,将套接字句柄分配给自己感兴趣的每个fd_set。
(3) 调用select函数,等待I/O操作的完成。
(4) 根据select的返回值,我们便可判断出哪些套接字存在着尚未完成(待决)的I/O操作,.具体的方法是使用FD_ISSET宏,对每个fd_set集合进行检查。
(5) 知道了每个集合中“待决”的I/O操作之后,对I/O进行处理,然后返回步骤1 ),继续进行select处理。
(6)select返回后,它会修改每个fd_set结构,删除那些不存在待决I/O操作的套接字句柄。这正是我们在上述的步骤( 4 )中,为何要使用FD_ISSET宏来判断一个特定的套接字是否仍在集合中的原因。

示例代码:

 1 SOCKET s;
 2 fd_set fdread;
 3 int ret;
 4
 5 //初始化
 6 ...
 7
 8 //Manage I/O on the socket
 9 while(1)
10 {
11     FD_ZERO(&fdread);//初始化集合
12
13     FD_SET(s,&fdread);//添加套接字到集合上
14
15     ret=select(0,&fdread,NULL,NULL,NULL);//调用select函数监视网络事件的完成情况
16     if(ret==SOCKET_ERROR)
17     {
18         ...
19     }
20
21     if(res>0)
22     {
23         //处理数据
24         ...
25
26         if(FD_ISSET(s,&fdread))//检查套接字是否还在集合上
27         {
28             ...
29         }
30     }
31
32 }

时间: 2024-11-08 09:51:31

一.Windows I/O模型之选择(select)模型的相关文章

C++ Windows 上简单的非阻塞Select模型

说明:当客户端连接数超过64个的时候,每次最多select64个,但每一帧可以select多次,理论上可以突破fd个数的限制 .h #ifndef _MODULE_SELECT_H_ #define _MODULE_SELECT_H_ #include "platform/platform.h" class CModuleSelect { public: CModuleSelect(); ~CModuleSelect(); public: int32_t Initialize(); i

Select模型

参考:http://m.blog.csdn.net/article/details?id=51420015 一.套接字模式 套接字模式简单的决定了操作套接字时,Winsock函数是如何运转的.Winsock以两种模式执行I/O操作:阻塞和非阻塞. 在阻塞模式下,执行I/0的Winsock调用(如send和recv)一直到操作完成才返回. 非阻塞模式下,Winsock函数会立刻返回 1.阻塞模式 套接字创建时,默认工作在阻塞模式下,列入对recv函数的调用会使程序进入等待状态,知道接收到数据才返回

Winsock—I/O模型之选择模型(一)

Winsock中提供了一些I/O模型帮助应用程序以异步方式在一个或多个套接字上管理I/O. 这样的I/O模型有六种:阻塞(blocking)模型,选择(select)模型,WSAAsyncSelect模型,WSAEventSelect模型,重叠(overlapped)模型,完成端口(completion port)模型. 选择模型: 目的:允许想要避免在套接字调用上阻塞的应用程序有能力管理多个套接字. 一.select函数             select函数可以确定一个或者多个套接字的状态

windows socket编程select模型使用

int select( int nfds,            //忽略 fd_ser* readfds,    //指向一个套接字集合,用来检测其可读性 fd_set* writefds,   // 指向一个套接字结合,用来检测其可写性 fd_ser* exceptfds, //指向一个套接字集合,用来检测错误 const struct timeval * timeout   //指定此函数等待的最长时间,如果为NULL,则最长时间为无限大. ); 参数说明: (1)   nfds  win

关于 Poco::TCPServer框架 (windows 下使用的是 select模型) 学习笔记.

说明 为何要写这篇文章 ,之前看过阿二的梦想船的<Poco::TCPServer框架解析> http://www.cppblog.com/richbirdandy/archive/2010/09/10/123994.html 无奈代码太多,看起繁琐.所以 准备 以流程图简化,便于理解.也方便自己以后使用. 本文内容 是基于window api分析的. 本文的poco是1.4.6p4 (2014-04-18)版本的. 虽然现在poco版本是1.6 但调用改动不大. poco下载地址:http:/

socket select模型

由于socket recv()方法是堵塞式的,当多个客户端连接服务器时,其中一个socket的recv调用时,会产生堵塞,使其他连接不能继续. 如果想改变这种一直等下去的焦急状态,可以多线程来实现(不再等待,同时去recv,同时阻塞),每个socket连接使用一个线程,这样效率十分低下,根本不可能应对负荷较大的情况(是啊,占用各种资源,电脑啊,你耗不起). 这时候我们便可以采取select模型.select允许进程指示内核等待多个事件中的任何一个发生,并仅在有一个或多个事件发生或经历一段指定时间

socket select()模型

转载:http://www.cnblogs.com/xiangshancuizhu/archive/2012/10/05/2711882.html 由于socket recv()方法是阻塞式的,当有多个客户端连接服务器时,其中一个socket的recv调用产生了阻塞,使其他链接不能继续.如果想改变这种一直等下去的焦急状态,可以多线程来实现(不再等待,同时去recv,同时阻塞),每个socket连接使用一个线程,这样效率十分低下,根本不可能应对负荷较大的情况(是啊,占用各种资源,电脑啊,你耗不起)

【转载】socket select模型

由于socket recv()方法是堵塞式的,当多个客户端连接服务器时,其中一个socket的recv调用时,会产生堵塞,使其他连接不能继续. 如果想改变这种一直等下去的焦急状态,可以多线程来实现(不再等待,同时去recv,同时阻塞,呵呵),每个socket连接使用一个线程,这样效率十分低下,根本不可能应对负荷较大的情况(是啊,占用各种资源,电脑啊,你耗不起). 这时候我们便可以采取select模型. select允许进程指示内核等待多个事件中的任何一个发生,并仅在有一个或多个时间发生或经历一段

select模型的原理、优点、缺点

关于I/O多路复用: I/O多路复用(又被称为“事件驱动”),首先要理解的是,操作系统为你提供了一个功能,当你的某个socket可读或者可写的时候,它可以给你一 个通知.这样当配合非阻塞的socket使用时,只有当系统通知我哪个描述符可读了,我才去执行read操作,可以保证每次read都能读到有效数据而不 做纯返回-1和EAGAIN的无用功.写操作类似.操作系统的这个功能通过select/poll/epoll之类的系统调用来实现,这些函数都可以同时 监视多个描述符的读写就绪状况,这样,**多个描