TCP併發ECHO服務器——多進程版

#include<stdio.h>
#include<string.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<pthread.h>
#include<unistd.h>
//TCP併發ECHO服務器(併發回執服務器----你給服務器發啥,服務器給你回啥)
void deal_client_fun(int fd)
{
	

	while(1)//服務器的核心代碼
	{
		//獲取客戶端請求
		char buf[128]="";
		int len = recv(fd,buf,sizeof(buf),0);
		if(len == 0)
			break;
		//迴應客戶端
		send(fd,buf,len,0);	
	}
	
	
}
int main()
{
	//創建套接字
	int sockfd = socket(AF_INET,SOCK_STREAM,0);
	if(sockfd<0)
	{
		perror("socket");
	}
	//端口複用
	int yes =1;
	setsockopt(sockfd,SOL_SOCKET,SO_REUSEADDR,&yes,sizeof(yes));
	//bind綁定
	struct sockaddr_in my_addr;
	bzero(&my_addr,sizeof(my_addr));
	my_addr.sin_family = AF_INET;
	my_addr.sin_port = htons(9000);
	my_addr.sin_addr.s_addr = htonl(INADDR_ANY);
	int ret = bind(sockfd,(struct sockaddr *)&my_addr,sizeof(my_addr));
	if(ret == -1)
	{
		perror("bind");
	}
	//listen創建連接隊列
	listen(sockfd,10);
	
	
	//accept從連接隊列提取已完成連接,得到已連接套接字
	////accept調用一次只能提取一個客戶端
	while(1)
	{
		struct sockaddr_in cli_addr;
		socklen_t cli_len=sizeof(cli_addr);
		int new_fd = accept(sockfd,(struct sockaddr *)&cli_addr,&cli_len);
		
		//遍歷客戶端信息ip
		char ip[16]="";
		
		inet_ntop(AF_INET,&cli_addr.sin_addr.s_addr,ip,16);
		unsigned short port=ntohs(cli_addr.sin_port);
		printf("已有客戶端:%s:%hu連接上了服務器\n",ip,port);
		
		pid_t pid;
		if(fork() == 0)//子進程
		{
			//關閉監聽套接字
			close(sockfd);
			
			deal_client_fun(new_fd);
			
			//關閉已連接套接字
			close(new_fd);
			_exit(0);
		}
		else//父進程
		{
			//監聽新的連接到來,不需要和客戶端通信 必須關閉已連接套接字
			close(new_fd);
		}
	}
	
	close(sockfd);
	
	return 0;
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章