select模型
在Linux中,我们可以使用select函数实现I/O端口的复用,传递给 select函数的参数会告诉内核:
1) 我们所关心的文件描述符
2) 对每个描述符,我们所关心的状态。(我们是要想从一个文件描述符中读或者写,还是关注一个描述符中是否出现异常)
3) 我们要等待多长时间。(我们可以等待无限长的时间,等待固定的一段时间,或者根本就不等待)
从select函数返回后,内核告诉我们一下信息:
1) 对我们的要求已经做好准备的描述符的个数
2) 对于三种条件哪些描述符已经做好准备.(读,写,异常)
返回:做好准备的文件描述符的个数,超时为0,错误为 -1.
最后一个参数。它指明我们要等待的时间:
struct timeval{
long tv_sec; /*秒 */
long tv_usec; /*微秒 */
}
有三种情况:
timeout == NULL 等待无限长的时间。等待可以被一个信号中断。当有一个描述符做好准备或者是捕获到一个信号时函数会返回。如果捕获到一个信号,
select函数将返回 -1,并将变量 erro设为 EINTR。
timeout->tv_sec == 0 &&timeout->tv_usec == 0不等待,直接返回。加入描述符集的描述符都会被测试,并且返回满足要求的描述符的个数。这种方法通过轮
询,无阻塞地获得了多个文件描述符状态。
timeout->tv_sec !=0 ||timeout->tv_usec!= 0 等待指定的时间。当有描述符符合条件或者超过超时时间的话,函数返回。在超时时间即将用完但又没有描述符
合条件的话,返回 0。对于第一种情况,等待也会被信号所中断。
函数返回:
1) 当监视的相应的文件描述符集中满足条件时,比如说读文件描述符集中有数据到来时,内核(I/O)根据状态修改文件描述符集,并返回一个大于0的数。
2) 当没有满足条件的文件描述符,且设置的timeval监控时间超时时,select函数会返回一个为0的值。
3) 当select返回负值时,发生错误。
server
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
#include <stdio.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <string.h> #include <unistd.h> int main(int argc, char** argv) { int listenfd1 = socket(AF_INET, SOCK_STREAM, 0); struct sockaddr_in serv_addr1; bzero((char*)&serv_addr1,sizeof(serv_addr1)); serv_addr1.sin_family = AF_INET; serv_addr1.sin_port = htons(7778); serv_addr1.sin_addr.s_addr = INADDR_ANY; bind(listenfd1, (struct sockaddr*)&serv_addr1,sizeof(serv_addr1)); listen(listenfd1,5); int maxfd = listenfd1; fd_set allset,rset; FD_ZERO(&allset); FD_SET(listenfd1,&allset); int clifd, clilen; struct sockaddr_in cli_addr; char buffer[256]; for(;;) { rset = allset; select(maxfd + 1, &rset, NULL, NULL, NULL); //如果是listenfd1 获取消息 if(FD_ISSET(listenfd1, &rset)) { clilen = sizeof(cli_addr); clifd = accept(listenfd1, (struct sockaddr *) &cli_addr, (socklen_t*)&clilen); bzero(buffer, 256); read(clifd, buffer, 255); printf("Listenfd1 Message is:%s\r\n", buffer); } } close(listenfd1); }
client
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
#include <stdio.h> #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <string.h> #include <unistd.h> int main(int argc, char* argv[]) { int socketfd; socketfd = socket(AF_INET,SOCK_STREAM,0); struct sockaddr_in serv_addr; bzero((char*)&serv_addr,sizeof(serv_addr)); serv_addr.sin_family = AF_INET; serv_addr.sin_port = htons(7778); connect(socketfd,(struct sockaddr*)&serv_addr,sizeof(serv_addr)); write(socketfd,"client_message",14); return 0; }