linux-----网络编程----select实现并发服务器

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<string.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<ctype.h>
#include<sys/select.h>
#include<netinet/in.h>

#define SERV_PORT 8080
 
int main()
{
	int lfd,cfd;
	struct sockaddr_in serv_addr,cli_addr;
	int serv_len,cli_len;
	
	//创建监听套接字:
	lfd =socket(AF_INET,SOCK_STREAM,0);
	
	//初始化服务器,协议,IP,端口相关:
	memset(&serv_addr,0,sizeof(serv_addr));
	serv_addr.sin_family = AF_INET;
	serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
	serv_addr.sin_port = htons(SERV_PORT);
	serv_len = sizeof(serv_addr);
	
	//绑定IP 和 端口:
	bind(lfd,(struct sockaddr*)&serv_addr,serv_len);
	
	//设置同时监听的最大个数:
	listen(lfd,36);
	//listen只做两件事:
	/*
	1,将socket创建的主动(默认)套接字转换成被动套接字,指示内核接受指向该套接字的连接请求
	2,指定内核应该为相应套接字排队的最大连接数
	*/
	printf("Waitting for connectomg...\n");
	
	//开始操作select相关的内容:
	int ret;
	int maxfd = lfd;//select函数中第一个参数,表示所有文件描述符最大值。文件描述符指向文件描述表,即表示其的最大值;

	//temp作为中间保存变量,保证现有连接能够被检测,并且下一次连接能够被保存。
	fd_set reads,temp;
	
	FD_ZERO(&reads);
	FD_ZERO(&temp);
	FD_SET(lfd,&reads);
	
	while(1){
		temp = reads;
		ret = select(maxfd+1,&temp,NULL,NULL,NULL);
		if(ret == -1){
			perror("select fail\n");
			exit(1);
		}
		
		//判断是否有新连接,如果有将新连接放入检测集合中:
		if(FD_ISSET(lfd,&temp)){
			//接受连接请求:
			cli_len = sizeof(cli_addr);
			cfd = accept(lfd,(struct sockaddr*)&cli_addr,&cli_len);
			
			//文件描述符放入检测集合:
			FD_SET(cfd,&reads);
			maxfd = maxfd>cfd?maxfd:cfd;
		}
		
		for(int i=lfd+1;i<=maxfd;i++){    //这个for循环的条件一定要注意!
			if(FD_ISSET(i,&temp)){
				//读数据:
				char buf[1024];
				int len = read(i,buf,sizeof(buf));
				if(len<0){
					perror("read fail\n");
					exit(1);
				}
				else if(len ==0){
					//此时对方i关i闭了连接:
					FD_CLR(i,&reads);
					close(i);
					if(maxfd==i){
						--maxfd;
					}
				}
				else{
					write(STDOUT_FILENO,buf,len);
				}
			}
		}
			
	}
	return 0;
}
发布了158 篇原创文章 · 获赞 37 · 访问量 3万+

猜你喜欢

转载自blog.csdn.net/XUCHEN1230/article/details/87108800