select模型

本文档详细分析了一个C语言实现的服务器端程序,着重讨论了如何解决fd_max参数未及时更新的问题。通过优化select和fd_set操作,确保在客户端连接和数据交换过程中,fd_max能够准确反映活跃套接字数量,提高程序效率。
摘要由CSDN通过智能技术生成
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <sys/select.h>

#define BUF_SIZE 100;
void error_handling(char *buf);

int main(char argc,char *argv[])
{
	int serv_sock,clnt_sock;
	struct sockaddr_in serv_adr,clnt_adr;
	struct timeval timeout;
	fd_set reads,cpy_reads;
	
	
	socklen_t adr_sz;
	int fd_max,str_len,fd_num,i;
	char buf[BUF_SIZE];
	if(argv != 2)
	{
		printf("Usage :%s <port> \n",argv[0]);
		exit(1);
	}
	
	serv_sock = socket(PF_INET,SOCK_STREAM,0);
	memset(&serv_adr,0,sizeof(serv_adr));
	serv_adr.sin_family = AF_INET;
	serv_adr.sin_port = htons(atoi(argv[1]));
	serv_adr.sin_addr.s_addr =htonl(INADDR_ANY);
	
	if(bind(serv_sock,(struct sockaddr*)&serv_adr,sizeof(serv_adr))== -1)
		error_handling("bind() error");
	if(listen(serv_adr,5) == -1)
		error_handling("listen() error");
	
	FD_ZERO(&read);
	FD_ISSET(serv_adr,&read);
	fd_max = serv_adr;
	
	while(1)
	{	
		coy_reads = reads;
		timeout.tv_sec = 5;
		timeout.tv_usec = 5000;
		
		if((fd_num = select(fd_max+1,&coy_reads,NULL,NULL,&timeout)) == -1)
			break;
		if(fd_num == 0)
			continue;
		
		for(i = 0; i < fd_max;i++)
		{
			if(FD_ISSET(i,&cpy_reads))
			{
				if(i == serv_sock)
				{
					adr_sz = sizeof(clnt_adr);
					clnt_sock = accept(serv_sock,(struct sockaddr*)&clnt_adr,&adr_sz);
					if(clnt_sock > fd_max)
						fd_max = clnt_sock;
					FD_SET(clnt_sock,&reads);
					printf("the clnt_sock is %d\n",clnt_sock);
				}else(
					str_len = read(i,buf,BUF_SIZE);
					if(str_len == 0)
					{
						FD_CLR(i,&reads);
						close(i);
						printf("close client : %d\n",i);
					}else{
						write(i,buf,str_len);
					}
				)
				
			}
		}
	}
	close(serv_sock);
	return 0;
}

void error_handling(char *buf)
{
	fputs(buf,stderr);
	fputs('\n',stderr);
	exit(1);
}

缺点与疑问:fd_max似乎没有即使的更新

流程梳理:

首先完成sock的操作socket()、bind()、listen();然后进入循环,将socket套接字交给select进行监视,这里要考虑到一些select的参数,因为在循环中对套接字进行监视,所以参数的部分是需要实时变化的

 int select(int nfds, fd_set *readfds, fd_set *writefds,
                  fd_set *exceptfds, struct timeval *timeout);
首先是select的阻塞问题,对参数struct timeval*timeout进行设置,大于0表示阻塞的时间,小雨0表示函数不阻塞,这个参数也是实时进行变化的,所以在进入select函数时需要重新设置时间进行更新(linux下)。

然后是:readfds参数,也需要实时的变化与赋值,将需要监视的套接字注册在readfds参数中,然后使用副本写入select函数,进行参数的传入和传出,和FD_ISSET的判断。

while()循环用来更新select的参数设置

然后使用for循环来判断套接字是否存在,如果存在是服务器的请求套接字,还是客户端的发送数据套接字。

如果是请求连接套接字:进行accept处理,将获得的clnt_sock注册到reads中去,将fd_max值进行更新。(更新select监视需要的参数)

如果是客户端套接字,则进行套接字处理,使用函数处理完成后需要关闭对应的套接字,并在reads将sock清除出去。

这个代码的缺点,没有在客户端的处理上没有对fd_max参数进行更新。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

潘多拉的面

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值