点击(此处)折叠或打开
- #include <sys/select.h>
- int select(int maxfdp1, fd_set *readfds, fd_set *writefds, fd_set *errorfds, const struct timeval *timeout);
点击(此处)折叠或打开
- maxfdp1: 待测试的文件描述字的个数,其值是待测试的最大描述符加1(由于描述符是从0开始索引)
- readfds: 指向对可读感兴趣的描述符集合的指针
- writefds: 指向对可写感兴趣的描述符集合的指针
- errorfds: 指向对处于异常条件感兴趣的描述符集合的指针
- timeout: 指定愿意等待的时间长度。若为NULL,则堵塞;如果其结构体内元素均为0,则直接非堵塞;如果以上两种均不满足,则等待固定时间后返回
点击(此处)折叠或打开
- FD_ZERO(fd_set *fdset); /* clear all bits in fdset */
- FD_SET(int fd, fd_set *fdset); /* turn on the bit for fd in fd_set */
- FD_CLR(int fd, fd_set *fdset); /* turn off the bit for fd in fd_set */
- int FD_ISSET(int fd, fd_set *fdset); /* is the bit for fd on in fdset? */
点击(此处)折叠或打开
- -1:出错
- 0:没有描述符准备好
- 正数:3个描述符集合当中已经准备好的描述符的个数之和
在redis当中,封装结构如下:
点击(此处)折叠或打开
- typedef struct aeApiState {
- fd_set rfds, wfds;
- /* We need to have a copy of the fd sets as it's not safe to reuse
- * FD sets after select(). */
- fd_set _rfds, _wfds;
- } aeApiState;
点击(此处)折叠或打开
- static int aeApiCreate(aeEventLoop *eventLoop) {
- aeApiState *state = zmalloc(sizeof(aeApiState));
- if (!state) return -1;
- FD_ZERO(&state->rfds);
- FD_ZERO(&state->wfds);
- eventLoop->apidata = state;
- return 0;
- }
- static int aeApiResize(aeEventLoop *eventLoop, int setsize) {
- /* Just ensure we have enough room in the fd_set type. */
- if (setsize >= FD_SETSIZE) return -1;
- return 0;
- }
- static void aeApiFree(aeEventLoop *eventLoop) {
- zfree(eventLoop->apidata);
- }
- static int aeApiAddEvent(aeEventLoop *eventLoop, int fd, int mask) {
- aeApiState *state = eventLoop->apidata;
- if (mask & AE_READABLE) FD_SET(fd,&state->rfds);
- if (mask & AE_WRITABLE) FD_SET(fd,&state->wfds);
- return 0;
- }
- static void aeApiDelEvent(aeEventLoop *eventLoop, int fd, int mask) {
- aeApiState *state = eventLoop->apidata;
- if (mask & AE_READABLE) FD_CLR(fd,&state->rfds);
- if (mask & AE_WRITABLE) FD_CLR(fd,&state->wfds);
- }
- static char *aeApiName(void) {
- return "select";
- }
点击(此处)折叠或打开
- static int aeApiPoll(aeEventLoop *eventLoop, struct timeval *tvp) {
- aeApiState *state = eventLoop->apidata;
- int retval, j, numevents = 0;
- memcpy(&state->_rfds,&state->rfds,sizeof(fd_set));
- memcpy(&state->_wfds,&state->wfds,sizeof(fd_set));
- retval = select(eventLoop->maxfd+1,
- &state->_rfds,&state->_wfds,NULL,tvp);
- if (retval > 0) {
- for (j = 0; j <= eventLoop->maxfd; j++) {
- int mask = 0;
- aeFileEvent *fe = &eventLoop->events[j];
- if (fe->mask == AE_NONE) continue;
- if (fe->mask & AE_READABLE && FD_ISSET(j,&state->_rfds))
- mask |= AE_READABLE;
- if (fe->mask & AE_WRITABLE && FD_ISSET(j,&state->_wfds))
- mask |= AE_WRITABLE;
- eventLoop->fired[numevents].fd = j;
- eventLoop->fired[numevents].mask = mask;
- numevents++;
- }
- }
- return numevents;
- }
由以上函数我们可以知晓,在ae_select设置的时候,使用的是rfds和wfds,在调用select的时候,是将上面两个拷贝到_rfds和_wfds当中,这里是为了避免重用描述符时的安全问题~
还望各位看官多提宝贵意见