本文实现一个多进程并发服务器
实现思路:
利用fork()函数实现每有一个client连接时就会创建一个子进程进程与client通信,父进程负责回收子进程的PCB
直接上代码!
#include <stdio.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <signal.h>
#include <sys/wait.h>
#include <ctype.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#define SERVER_PORT 8888
#define MAXLINE 8192
//进程回收函数
void do_sigchlid(int num)
{
pid_t pid;
while((pid = waitpid(-1, NULL, WNOHANG)) > 0);//回收任何子进程
{
printf("child died, pid = %d\n", pid);
}
}
int main()
{
int listenfd, connfd;
pid_t pid;
int len;
char buf[MAXLINE];
char str[256];
int i;
//创建监听描述符
listenfd = socket(AF_INET, SOCK_STREAM, 0);
if(listenfd == -1)
{
perror("socket error");
exit(1);
}
//she值端口复用
int opt = 1;
setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
//绑定ip和端口
struct sockaddr_in serveraddr;
bzero(&serveraddr, sizeof(serveraddr));
serveraddr.sin_port = htons(SERVER_PORT);
serveraddr.sin_family = AF_INET;
serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);
int ret = bind(listenfd, (struct sockaddr*)&serveraddr, sizeof(serveraddr));
if(ret == -1)
{
perror("bind error");
exit(1);
}
//设置监听
ret = listen(listenfd, 20);
if(ret == -1)
{
perror("listen error");
exit(1);
}
printf("Accepting connections ...\n");
struct sockaddr_in clientaddr;
socklen_t clientaddr_len = sizeof(clientaddr);
//通信
while(1)
{
//等待接收客户端的连接
connfd = accept(listenfd, (struct sockaddr*)&clientaddr, &clientaddr_len);
printf("connected client ip:%s, port: %d\n",
inet_ntop(AF_INET, &clientaddr.sin_addr, str, sizeof(str)), ntohs(clientaddr.sin_port));
//有一个新的连接就创建一个进程
pid = fork();
if(pid == 0)
{
//子进程
//关闭监听文件描述符
close(listenfd);
while(1)//与客户端通信,机型数据处理
{
//读数据
len = read(connfd, buf, MAXLINE);
if(len == 0)
{
printf("客户端断开了连接...\n");
break;
}
for(i = 0; i < len; ++i)
{
buf[i] = toupper(buf[i]);
}
//写回客户端
write(STDOUT_FILENO, buf, len);//把buf写到标准输出中去
write(connfd, buf, len);//把buf写回客户端
}
close(connfd);
return 0;
}
else if(pid > 0)
{
//父进程,负责对子进程pcb资源的回收
//使用信号来回收,子进程结束会给父进程发送SIGCHLD信号
//默认父进程会屏蔽此信号,若父进程想回收子进程资源,
//可以捕捉此信号
//信号捕捉
//注册捕捉函数
struct sigaction act;
act.sa_flags = 0;
act.sa_handler = do_sigchlid;//回调函数,通过sa_handler指针调用
sigemptyset(&act.sa_mask);//将信号集初始化为空
sigaction(SIGCHLD, &act, NULL);
}
}
return 0;
}