fd_set结构体
在Linux中,内核利用文件描述符(File Descriptor)即文件句柄,来访问文件。文件描述符在形式上是一个非负整数。实际上,它是一个索引值,指向内核为每一个进程所维护的该进程打开文件的记录表。打开现存文件或新建文件时,内核会返回一个文件描述符。读写文件也需要使用文件描述符来指定待读写的文件。宏FD_ZERO、FD_SET、FD_CLR、FD_ISSET中“FD”即为file descriptor的缩写,下面来一一进行介绍。
首先介绍一个重要的结构体:fd_set,它会作为下面某些函数的参数而多次用到,fd_set可以理解为一个集合,这个集合中存放的是文件描述符(file descriptor),即文件句柄。
在/usr/include/sys/select.h中有:
typedef long int __fd_mask;
/* It's easier to assume 8-bit bytes than to get CHAR_BIT. */
#define __NFDBITS (8 * (int) sizeof (__fd_mask))
#define __FDELT(d) ((d) / __NFDBITS)
#define __FDMASK(d) ((__fd_mask) 1 << ((d) % __NFDBITS))
/* fd_set for select and pselect. */
typedef struct
{
/* XPG4.2 requires this member name. Otherwise avoid the name
from the global namespace. */
#ifdef __USE_XOPEN
__fd_mask fds_bits[__FD_SETSIZE / __NFDBITS];
# define __FDS_BITS(set) ((set)->fds_bits)
#else
__fd_mask __fds_bits[__FD_SETSIZE / __NFDBITS];
# define __FDS_BITS(set) ((set)->__fds_bits)
#endif
} fd_set;
/* Maximum number of file descriptors in `fd_set'. */
#define FD_SETSIZE __FD_SETSIZE //__FD_SETSIZE等于1024
/* Access macros for `fd_set'. */
#define FD_SET(fd, fdsetp) __FD_SET (fd, fdsetp)
#define FD_CLR(fd, fdsetp) __FD_CLR (fd, fdsetp)
#define FD_ISSET(fd, fdsetp) __FD_ISSET (fd, fdsetp)
#define FD_ZERO(fdsetp) __FD_ZERO (fdsetp)
从上面的代码中简化下来,其实可以改成:
typedef struct{
long int fds_bits[32];
}fd_set;
其实fd_set就是一个long int类型的数组。因为每一位可以代表一个文件描述符,所以fd_set最多表示1024个文件描述符。
常用宏操作
fd_set集合可以通过下面的宏来进行人为来操作:FD_ZERO:用来清空fd_set集合,即让fd_set集合不再包含任何文件句柄。
FD_SET:用来将一个给定的文件描述符加入集合之中
FD_CLR:用来将一个给定的文件描述符从集合中删除
FD_ISSET:检测fd在fdset集合中的状态是否变化,当检测到fd状态发生变化时返回真,否则,返回假(也可以认为集合中指定的文件描述符是否可以读写)。
函数select()的用法
函数原型:
int select(int maxfdp,fd_set *readfds,fd_set *writefds,fd_set *errorfds,struct timeval *timeout);
作用:用来够监视我们需要监视的文件描述符(读或写的文件集中的文件描述符)的状态变化情况。并能通过返回的值告知我们。
返回值:
>0:被监视的文件描述符有变化
-1:出错
0 :超时
参数解释:
int maxfdp:集合中所有文件描述符的范围,为所有文件描述符的最大值加1。
fd_set *readfds:要进行监视的读文件集。
fd_set *writefds :要进行监视的写文件集。
fd_set *errorfds:用于监视异常数据。
struct timeval* timeout:select的超时时间,它可以使select处于三种状态:
第一,若将NULL以形参传入,即不传入时间结构,就是 将select置于阻塞状态,一定等到监视文件描述符集合中某个文件描述符发生变化为止;
第二,若将时间值设为0秒0毫秒,就变成一个纯粹的非阻塞函数, 不管文件描述符是否有变化,都立刻返回继续执行,文件无变化返回0,有变化返回一个正值;
第三,timeout的值大于0,这就是等待的超时时间,即 select在timeout时间内阻塞,超时时间之内有事件到来就返回了,否则在超时后不管怎样一定返回。
struct timeval timeout; timeout.tv_sec = 0; //秒 timeout.tv_usec = dwTimeout * 1000; //微秒 1毫秒 = 1000微秒
参考文献:
[1]. https://my.oschina.net/floristgao/blog/311612
[2]. http://m.blog.chinaunix.net/uid-30271883-id-5604817.html