我们看到tcp客户同时处理两个输入:标准输入和tcp套接口。遇到的问题是就在客户阻塞于(标准输入上的)fgets调用期间,服务器进程被杀死。服务器tcp虽然正确的给客户tcp发送了一个FIN,但是既然客户进程正阻塞于从标准输入读入,它将看不到这个EOF,直到从套接口读时为止(可能已经过了很长一段时间)。这样的进程需要一种预先告知内核的能力,使得内核一旦发现进程指定的一个或者多个I/O条件就绪(也就是说输入已准备好被读取,或者描述字已能承接更多的输出),它就通知进程。这个能力称为I/O复用(I/O multiplexing),是由select和poll这两个函数支持的。
I/O复用典型使用在下列网络应用场合:
1).当处理多个描述字(通常是交互式输入和网络套接口)时,必须使用I/O复用。
2).一个客户同时处理多个套接口是可能的。
3).如果一个tcp服务器既要处理监听套接口,又要处理已连接套接口,一般就要使用I/O复用
4).如果一个服务器既要处理tcp,又要处理udp,一般需要使用I/O复用。
5).如果一个服务器要处理多个服务或者多个协议,一般就要使用I/O复用。
select函数:
该函数允许进程指示内核等待多个事件中的任何一个发生,并仅在有一个或多个事件发生或经历一段指定的时间后才唤醒它。
select函数原型:
#include
#include
int select(int maxfdp1, fd_set *readset, fd_set *writeset,
fd_set *exceptset, const struct timeval *timeout);
//返回:就绪描述字的正数目。0--超时, -1---出错
struct timeval {
long tv_sec;
long tv_usec;
};
4个参数的解释:
maxfdp1:指定待测试的描述字个数,它的值是待测试的最大描述字加1.描述字0、1、2.....一直到maxfdp1-1均被测试。这个参数存在的意义:为了效率的原因。每个fd_set都有表示大量描述字的空间,然而一个普通进程所用的数量却少得多。内核正是通过在进程与内核之间不拷贝描述字集中不必要的部分,从而不测试总为0的那些位来获得效率的。
readset、writeset、exceptset指定我们让内核测试读、写和异常条件的描述字。如何让这3个参数的每一个指定一个或者多个描述字是一个设计上的问题。select使用描述字集,典型的是一个整数数组,其中每个整数中的每一位对应一个描述字。
void FD_ZERO(fd_set *set); //clear all bits in set
void FD_SET(int fd, fd_set *set); //turn on the bit for fd in set
void FD_CLR(int fd, fd_set *set); //turn off the bit for fd in set
int FD_ISSET(int fd, fd_set * set); //is the bit for fd on in set?
我们分配一个fd_set数据类型的描述字集,并用这些宏设置或测试该集合中的每一位。
timeout:这个参数有3种可能:
1).永远等待下去。仅在有一个描述字准备好I/O时才返回。为此,我们把参数设置为NULL.
2).等待一段时间:在有一个描述字返回准备好I/O时才返回,但是不超过由该参数所指向的timeval结构中指定的秒数和微秒数。
3).根本不等待:检查描述字后立即返回,这称为论询(polling).为此,该参数必须指向一个timeval结构,而且其中的定时器值必须为0.
该函数返回时,结果指示哪些描述字已就绪。该函数返回后,使用FD_ISSET宏来测试fd_set数据类型中的描述字。描述字中任何与未就绪的描述字对应的位返回时均清0.为此,每次重新调用select时,我们都得再次把所有描述字集中所关心的位均置为0.
使用select时最常见的两个编程错误:忘了对最大描述字加1;忘了描述字集是值-结果参数。第二个错误导致调用select时,描述字集中我们认为是1的位却被置为0.
早先的版本在于:当套接口上发生某些事件时,客户可能阻塞与fgets调用。新版本改为阻塞于select调用,等待要么标准输入可读,要么套接口可读。
void str_cli(FILE *fp, int sockfd) {
int maxfdp1;
fd_set rset;
char send[MAXLINE], recv[MAXLINE];
FD_ZERO(&rset);
for(;;) {
FD_SET(fileno(fp), &rset);
FD_SET(sockfd, &rset);
maxfdp1 = max(fileno(fp), sockfd) + 1;
/*
* fileno函数把标准I/O文件指针转换为对应的描述字。select只工作在描述字上。
*/
select(maxfdp1, &rset, NULL, NULL, NULL);
if(FD_ISSET(sockfd, &rset)) {
if(0 == readline(sockfd, recv, MAXLINE))
perror("xxxx");
fputs(recv, stdout);
}
if(FD_ISSET(fileno(fp), &rset)) {
if(fgets(send, MAXLINE, fp) == NULL)
return;
write(sockfd, send, strlen(send));
}
}
}