linux(一)------多进程并发服务器实现(fork)

本文实现一个多进程并发服务器
实现思路:
利用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;
}

这里写图片描述

猜你喜欢

转载自blog.csdn.net/tuoniaoatuoniao/article/details/76549519