使用fork并发处理多个client的请求和对等通信p2p

来源:互联网 发布:淘宝卖家评论回复语 编辑:程序博客网 时间:2024/06/05 04:00

一、在前面讲过的回射客户/服务器程序中,服务器只能处理一个客户端的请求,如何同时服务多个客户端呢?在未讲到select/poll/epoll等高级IO之前,比较老土的办法是使用fork来实现。网络服务器通常用fork来同时服务多个客户端,父进程专门负责监听端口,每次accept一个新的客户端连接就fork出一个子进程专门服务这个客户端。但是子进程退出时会产生僵尸进程,父进程要注意处理SIGCHLD信号和调用wait清理僵尸进程,最简单的办法就是直接忽略SIGCHLD信号。

#include<stdio.h>#include<sys/types.h>#include<sys/socket.h>#include<unistd.h>#include<stdlib.h>#include<errno.h>#include<arpa/inet.h>#include<netinet/in.h>#include<string.h>#include<signal.h>#define ERR_EXIT(m) \    do { \        perror(m); \        exit(EXIT_FAILURE); \    } while (0)void do_service(int);int main(void){    signal(SIGCHLD, SIG_IGN);    int listenfd; //被动套接字(文件描述符),即只可以accept, 监听套接字    if ((listenfd = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP)) < 0)        //  listenfd = socket(AF_INET, SOCK_STREAM, 0)        ERR_EXIT("socket error");    struct sockaddr_in servaddr;    memset(&servaddr, 0, sizeof(servaddr));    servaddr.sin_family = AF_INET;    servaddr.sin_port = htons(5188);    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);    /* servaddr.sin_addr.s_addr = inet_addr("127.0.0.1"); */    /* inet_aton("127.0.0.1", &servaddr.sin_addr); */    int on = 1;    if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) < 0)        ERR_EXIT("setsockopt error");    if (bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0)        ERR_EXIT("bind error");    if (listen(listenfd, SOMAXCONN) < 0) //listen应在socket和bind之后,而在accept之前        ERR_EXIT("listen error");    struct sockaddr_in peeraddr; //传出参数    socklen_t peerlen = sizeof(peeraddr); //传入传出参数,必须有初始值    int conn; // 已连接套接字(变为主动套接字,即可以主动connect)    pid_t pid;    while (1)    {        if ((conn = accept(listenfd, (struct sockaddr *)&peeraddr, &peerlen)) < 0) //3次握手完成的序列            ERR_EXIT("accept error");        printf("recv connect ip=%s port=%d\n", inet_ntoa(peeraddr.sin_addr),               ntohs(peeraddr.sin_port));        pid = fork();        if (pid == -1)            ERR_EXIT("fork error");        if (pid == 0)        {            // 子进程            close(listenfd);            do_service(conn);            exit(EXIT_SUCCESS);        }        else            close(conn); //父进程    }    return 0;}void do_service(int conn){    char recvbuf[1024];    while (1)    {        memset(recvbuf, 0, sizeof(recvbuf));        int ret = read(conn, recvbuf, sizeof(recvbuf));        if (ret == 0)   //客户端关闭了        {            printf("client close\n");            break;        }        else if (ret == -1)            ERR_EXIT("read error");        fputs(recvbuf, stdout);        write(conn, recvbuf, ret);    }}


二、在最基本的回射客户/服务器程序中,服务器只能被动接收客户端的信息,而不能主动发送信息给客户端,如果我们想实现对等通信,即P2P,可以

在服务器程序用使用两个进程,一个进程接收用户的输入并发送给客户端,另一个进程被动接收客户端的消息并打印出来,此进程当read 返回0 时得知

客户端已经关闭需要退出进程,此时尚有另一个进程未退出,可以通过在退出前发送消息给它,在消息处理函数中退出。当然客户端也必须使用双进

程,原理与服务器程序相同。

#include<stdio.h>#include<sys/types.h>#include<sys/socket.h>#include<unistd.h>#include<stdlib.h>#include<errno.h>#include<arpa/inet.h>#include<netinet/in.h>#include<string.h>#include<signal.h>#define ERR_EXIT(m) \    do { \        perror(m); \        exit(EXIT_FAILURE); \    } while (0)void handler(int sig){    printf("recv a sig=%d\n", sig);    exit(EXIT_SUCCESS);}int main(void){    int listenfd; //被动套接字(文件描述符),即只可以accept    if ((listenfd = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP)) < 0)        //  listenfd = socket(AF_INET, SOCK_STREAM, 0)        ERR_EXIT("socket error");    struct sockaddr_in servaddr;    memset(&servaddr, 0, sizeof(servaddr));    servaddr.sin_family = AF_INET;    servaddr.sin_port = htons(5188);    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);    /* servaddr.sin_addr.s_addr = inet_addr("127.0.0.1"); */    /* inet_aton("127.0.0.1", &servaddr.sin_addr); */    int on = 1;    if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) < 0)        ERR_EXIT("setsockopt error");    if (bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0)        ERR_EXIT("bind error");    if (listen(listenfd, SOMAXCONN) < 0) //listen应在socket和bind之后,而在accept之前        ERR_EXIT("listen error");    struct sockaddr_in peeraddr; //传出参数    socklen_t peerlen = sizeof(peeraddr); //传入传出参数,必须有初始值    int conn; // 已连接套接字(变为主动套接字,即可以主动connect)    if ((conn = accept(listenfd, (struct sockaddr *)&peeraddr, &peerlen)) < 0)        ERR_EXIT("accept error");    printf("recv connect ip=%s port=%d\n", inet_ntoa(peeraddr.sin_addr),           ntohs(peeraddr.sin_port));    pid_t pid;    pid = fork();    if (pid == -1)        ERR_EXIT("fork error");    if (pid == 0)    {        signal(SIGUSR1, handler);        char sendbuf[1024] = {0};        while (fgets(sendbuf, sizeof(sendbuf), stdin) != NULL)        {            write(conn, sendbuf, strlen(sendbuf));            memset(sendbuf, 0, sizeof(sendbuf));        }        exit(EXIT_SUCCESS);    }    else    {        char recvbuf[1024];        while (1)        {            memset(recvbuf, 0, sizeof(recvbuf));            int ret = read(conn, recvbuf, sizeof(recvbuf));            if (ret == -1)                ERR_EXIT("read error");            else if (ret == 0)            {                printf("peer close\n");                break;            }            fputs(recvbuf, stdout);        }        kill(pid, SIGUSR1); //父进程退出时发送信号给子进程        exit(EXIT_SUCCESS);    }}


0 0
原创粉丝点击