#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;
}
TCP併發ECHO服務器——多進程版
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.