用Socket技术,完成一个简单的并发服务器,要求用fork()创建子进程来并发处理实现多进程并发,在接收到请求时,fork个进程,对父进程则关闭连接Socket,而子进程则关闭监听socket,每一个新的accept请求都由子进程来执行,而由父进程继续listen。
server.c 端
#include<stdio.h>
#include<stdlib.h>#include<strings.h>
#include<string.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#define PORT 1234
#define BACKLOG 2 //numbers of allowed connections
#define MAXDATASIZE 1000
void process_cli(int connectfd,struct sockaddr_in client);
void main()
{
int listenfd,connectfd;
pid_t pid;
struct sockaddr_in server;
struct sockaddr_in client;
int sin_size;
//创建socket关键字
listenfd = socket(AF_INET,SOCK_STREAM,0);
if(listenfd == -1)
{
perror("creating socket failed");
exit(1);
}
int opt = SO_REUSEADDR;
setsockopt(listenfd,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(opt));
bzero(&server,sizeof(server));
server.sin_family = AF_INET;
server.sin_port = htons(PORT);
server.sin_addr.s_addr = htonl(INADDR_ANY);
//绑定端口和IP地址
if((bind(listenfd,(struct sockaddr *)&server,sizeof(struct sockaddr)))== -1)
{
perror("bind error\n");
exit(1);
}
//监听客户端
if(listen(listenfd,BACKLOG)==-1)
{
perror("listen error\n");
exit(1);
}
sin_size = sizeof(struct sockaddr_in);
while(1)
{
//等待接收来自客户端的信息
if((connectfd=accept(listenfd,(struct sockaddr*)&client,&sin_size))==-1)
{
perror("accept failed\n");
exit(1);
}
//创建进程
if((pid=fork())>0)
{
close(connectfd);
continue;
}
else if(pid ==0)
{
close(listenfd);
process_cli(connectfd,client);
exit(1);
}
else
{
printf("fork error\n");
exit(0);
}
}
close(listenfd);
}
void process_cli(int connectfd,struct sockaddr_in client)
{
int num;
char recvbuf[MAXDATASIZE],sendbuf[MAXDATASIZE],cli_name[MAXDATASIZE];
printf("you got a connectfd from %s\n.",inet_ntoa(client.sin_addr));
send(connectfd,"welcome to my server.\n",22,0);
num=recv(connectfd,cli_name,MAXDATASIZE,0);
if(num == 0)
{
close(connectfd);
printf("client connects failed\n");
return ;
}
cli_name[num-1]='\0';
printf("client'name is %s\n",cli_name);
while(num = recv(connectfd,recvbuf,MAXDATASIZE,0))
{
int i=0;
recvbuf[num]='\0';
printf("recevied client(%s)message:%s\n",cli_name,recvbuf);
for(i=0;i<num;i++)
{
sendbuf[i] = recvbuf[num-i-2];
}
sendbuf[num-1]='\0';
send(connectfd,sendbuf,strlen(sendbuf),0);
}
close(connectfd);
}
client.c 端
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<netdb.h>
#define PORT 1234
#define MAXDATASIZE 1000
int main(int argc,char *argv[])
{
int fd,numbytes;
char buf[MAXDATASIZE];
struct hostent *he;
struct sockaddr_in server;
int i=1;
if((he=gethostbyname(argv[1]))==NULL)
{
perror("gethostbyname error");
exit(1);
}
if((fd=socket(AF_INET,SOCK_STREAM,0))==-1)
{
perror("socket failed");
exit(1);
}
bzero(&server,sizeof(server));
server.sin_family =AF_INET;
server.sin_port =htons(PORT);
server.sin_addr =*((struct in_addr*)he->h_addr);
if(connect(fd,(struct sockaddr*)&server,sizeof(struct sockaddr))==-1)
{
perror("bind error");
exit(1);
}
if((numbytes=recv(fd,buf,MAXDATASIZE,0))==-1)
{
perror("recv error");
exit(1);
}
buf[numbytes]='\0';
printf("%s\n",buf);
printf("input your name:");
scanf("%s",buf);
if((numbytes=send(fd,buf,strlen(buf),0))==-1)
{
perror("send failed");
exit(1);
}
while(i)
{
printf("input me message(max char:%d):",MAXDATASIZE);
scanf("%s",buf);
if(strlen(buf)<1)
{
i=0;
}
if((numbytes=send(fd,buf,strlen(buf),0))==-1)
{
perror("send failed");
exit(1);
}
if((numbytes=recv(fd,buf,MAXDATASIZE,0))==-1)
{
perror("recv error");
exit(1);
}
buf[numbytes]='\0';
printf("server message:%s\n",buf);
printf("\n");
}
close(fd);
return 0;
}
运行结果可以多客户端对应一个服务器端。