我们明白了文件描述符集的实现机制之后,就可对其进行灵活运用。(以下程序在红帽Linux 6.0下运行通过,函数fd_isempty用于判断文件描述符集是否为空;函数fd_fetch取出文件描述符集中的所有文件描述符)
#include <stdio.h> #include <string.h> #include <sys/time.h> #include <sys/select.h> struct my_fd_set{ fd_set fs; //定义文件描述符集fs unsigned int nconnect; //文件描述符集fs中文件描述符的个数 unsigned int nmaxfd; //文件描述符集fs中最大的文件描述符 }; /* 函数fd_isempty用于判断文件描述符集是否为空,为空返回1,不为空则返回0 */ int fd_isempty(struct my_fd_set *pfs) { int i; /* 文件描述符集fd_set是通过整数数组来实现的,所以定义整数数组myset的元素个数为文件描述符集fd_set所占内存空间的字节数除以整数所占内存空间的字节数。 */ unsigned int myset[sizeof(fd_set) / sizeof(int)]; /* 把文件描述符集pfs->fs 拷贝到数组myset */ memcpy(myset, &pfs->fs, sizeof(fd_set)); for(i = 0; i < sizeof(fd_set) / sizeof(int); i++) /* 如果myset的某个元素不为0,说明文件描述符集不为空,则函数返回0 */ if (myset[i]) return 0; return 1; /* 如果myset的所有元素都为0,说明文件描述符集为空,则函数返回1 */ } /* 函数fd_fetch对文件描述符集进行位操作,把为1的位换算成相应的文件描述符,然后就可对其进行I/O操作 */ void fd_fetch(struct my_fd_set *pfs) { struct my_fd_set *tempset; //定义一个临时的结构指针 unsigned int myset[sizeof(fd_set)/sizeof(unsigned int)]; unsigned int i, nbit, nfind, ntemp; tempset = pfs; memcpy(myset, &tempset->fs, sizeof(fd_set)); /* 把最大的文件描述符maxfd除以整数所占的位数,得出maxfd在文件描述符集中相应的位对应于整数数组myset的相应元素的下标,目的是为了减少检索的次数 */ nfind = tempset->nmaxfd / (sizeof(int)*8); for (i = 0; i <= nfind; i++) { /* 如果数组myset的某个元素为0,说明这个元素所对应的文件描述符集的32位全为0,则继续判断下一元素。*/ if (myset[i] == 0) continue; /* 如果数组myset的某个元素不为0,说明这个元素所对应的文件描述符集的32位中有为1的,把myset[i]赋值给临时变量ntemp,对ntemp进行位运算,把为1的位换算成相应的文件描述符 */ ntemp = myset[i]; /* nbit记录整数的二进制位数,对ntemp从低到高位进行&1运算,直到整数的最高位,或直到文件描述符集中文件描述符的个数等于0 */ for (nbit = 0; tempset->nconnect && (nbit < sizeof(int)*8); nbit++) { if (ntemp & 1) { /* 如果某位为1,则可得到对应的文件描述符为nbit + 32*I,然后我们可对其进行I/O操作。这里我只是做了简单的显示。*/ printf("i = %d, nbit = %d, The file description is %d ", i, nbit, nbit + 32*i); /* 取出一个文件描述符后,将文件描述符集中文件描述符的个数减1 */ tempset->nconnect--; } ntemp >>= 1; // ntemp右移一位 } } } /* 下面的主程序是对以上两个函数的测试 */ main() { /* 假设fd1,fd2,fd3为3个文件描述符,实际运用中可为Socket描述符等 */ int fd1 = 7, fd2 = 256, fd3 = 1023, isempty; struct my_fd_set connect_set; connect_set.nconnect = 0; connect_set.nmaxfd = 0; FD_ZERO(&connect_set.fs); /* FD_SET操作前对函数fd_isempty进行测试 */ isempty = fd_isempty(&connect_set); printf("isempty = %d ", isempty); FD_SET(fd1, &connect_set.fs); FD_SET(fd2, &connect_set.fs); FD_SET(fd3, &connect_set.fs); connect_set.nconnect = 3; connect_set.nmaxfd = fd3 ; /* FD_SET操作后,既把文件描述符加入到文件描述符集之后,对函数fd_isempty进行测试 */ isempty = fd_isempty(&connect_set); printf("isempty = %d ", isempty); /* 对函数fd_ fetch进行测试 */ fd_fetch(&connect_set); }
/* 程序输出结果为 :*/ isempty is 1 isempty is 0 i = 0, nbit = 7, The file description is 7 i = 8, nbit = 0, The file description is 256 i = 31, nbit = 31, The file description is 1023