#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服务器——多进程版
最新推荐文章于 2021-11-30 14:57:35 发布