select函数、I/O复用、并发服务器
select函数原型
c
extern int select (int __nfds, fd_set *__restrict __readfds,
fd_set *__restrict __writefds,
fd_set *__restrict __exceptfds,
struct timeval *__restrict __timeout);
参数介绍:
- int __nfds:
- 这是需要监视的文件描述符集合中最大的文件描述符值加1。因为文件描述符是从0开始分配的,所以如果你有文件描述符3,那么你实际上是在监视0, 1, 2, 和3这四个文件描述符(尽管可能其中一些并没有被使用)。因此,这个参数应该设置为4(即最大文件描述符+1)。
操作fe_set集合统一的宏来处理:
• FD_SET(fd, &set): 将文件描述符fd添加到set集合中。
• FD_CLR(fd, &set): 从set集合中删除文件描述符fd。
• FD_ISSET(fd, &set): 检查fd是否在set集合中。
• FD_ZERO(&set): 清空set集合。
- fd_set *__restrict __readfds:
- 这是一个指向文件描述符集合的指针,该集合中的文件描述符被监视以查看它们是否可读。如果不需要监视任何文件描述符是否可读,这个参数可以设置为NULL。
- fd_set *__restrict __writefds:
- 类似于 __readfds,这个参数指向一个文件描述符集合,集合中的文件描述符被监视以查看它们是否可写。如果不需要监视任何文件描述符是否可写,这个参数可以设置为NULL。
- fd_set *__restrict __exceptfds:
- 这也是一个指向文件描述符集合的指针,集合中的文件描述符被监视以查看是否有异常条件发生(比如带外数据到达)。如果不需要监视任何文件描述符的异常条件,这个参数可以设置为NULL。
- struct timeval *__restrict __timeout:
- 这是一个指向 timeval 结构体的指针,该结构体指定了 select 函数等待的最长时间。如果设置为NULL,select 将无限期地等待,直到某个文件描述符准备好。
- timeval 结构体通常包含两个成员:tv_sec(秒)和 tv_usec(微秒),用于指定时间间隔。
- 返回值:
- select 函数返回准备好的文件描述符的总数(即可读、可写或有异常条件的文件描述符总数)。
- 如果返回-1,则表示发生了错误(此时应该检查 errno 来确定错误的具体原因)。
- 如果返回0,这表示超时。
可以直接看使用实例:可以使用任意客户端代码进行测试
c
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<netinet/in.h>
#include<sys/socket.h>
#include<sys/select.h>
#define LISTEN_NUM 10
void error_handing(char *message)
{
fputs(message,stderr);
fputc('\n',stderr);
exit(1);
}
int main(int argc, char const *argv[])
{
//步骤1:创建套接字
int server_sock;
server_sock = socket(PF_INET,SOCK_STREAM,IPPROTO_TCP);
if (server_sock == -1)
{
error_handing("socket() error");
}
//步骤2:绑定监听端口
struct sockaddr_in server_addr;
memset(&server_addr,0,sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
server_addr.sin_port = htons(atoi("3333"));
if (bind(server_sock,(struct sockaddr *)&server_addr,sizeof(server_addr)) == -1)
{
error_handing("bind() error");
}
//步骤3:开启监听
if (listen(server_sock,LISTEN_NUM) == -1)
{
error_handing("listen() error");
}
//步骤4:做select调用之前的准备
//步骤4.1:初始化sock 集合
fd_set fd_reads,fd_copy;//经过select函数之后,需要比较fd_set中的位是否改变,所以这里总是需要两个fd_set变量
FD_ZERO(&fd_reads);
//步骤4.2:将server_sock加入到集合中
FD_SET(server_sock,&fd_reads);
//步骤4.3:初始化最大值
int fd_max = server_sock;//定义最大的socket
while (1)
{
fd_copy = fd_reads;
//步骤4.4:设置超时时间
struct timeval time_out;
time_out.tv_sec = 5;
//步骤5:调用select函数,讨论其返回值
int activity_sockets = select(fd_max + 1,&fd_copy,NULL,NULL,&time_out);
if (activity_sockets == -1)//发生错误,退出
{
error_handing("select() error");
}
else if (activity_sockets == 0)//超时
{
//选择继续运行
printf("Time out.\n");
continue;
}
for (size_t i = 0; i < fd_max + 1; i++)//比较是否对应的事件位是否改变
{
if (FD_ISSET(i,&fd_copy))//说明i事件活动了
{
if(i == server_sock)//说明有新的客户端接入
{
printf("New client connected...\n");
struct sockaddr client_addr;
int sizeof_client_addr = sizeof(client_addr);
int client_sock = accept(server_sock,&client_addr,&sizeof_client_addr);
//将客户端的socket加入到总的fd_set中,就可以使用select查询处理
FD_SET(client_sock,&fd_reads);
//更新最大值,因为select监听的sockt是从0~fd_max
fd_max = (client_sock > fd_max) ? client_sock : fd_max;
}
else//说明有客户端数据进来
{
const int message_size = 30;
char message[message_size];
int read_size;
read_size = read(i,message,message_size);
if (read_size == 0)//说明客户端请求关闭连接
{
printf("Close client...\n");
FD_CLR(i,&fd_reads);//踢出监控集合
close(i);//再关闭连接
}
else//收到数据,选择给它回发过去
{
message[read_size] = '\0';
printf("Receive message is %s.\n",message);
write(i,message,read_size);
}
}
}
}
}
close(server_sock);
return 0;
}