TCP并发服务器的多进程实现与多线程实现

TCP并发服务器的多进程实现与多线程实现

一、 TCP并发服务器的多进程实现

代码

c 复制代码
#include <my_head.h>

#define SERVER_IP "192.168.125.11" //  服务器IP
#define SERVER_PORT 6666           //  服务器端口

// 子进程处理客户端信息函数
int deal_client_message(int new_sfd, struct sockaddr_in client_info);
// 回调函数回收僵尸进程函数
void callBack_zembie(int sig);

int main(int argc, const char *argv[])
{
    // 用信号的方式回收僵尸进程
    if (signal(SIGCHLD, callBack_zembie) < 0)
    {
        ERR_MSG("signal");
        return -1;
    }

    // 创建流式套接字
    int sfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sfd < 0)
    {
        ERR_MSG("socket");
        return -1;
    }

    // 待绑定的服务器的信息
    struct sockaddr_in server_info;
    server_info.sin_addr.s_addr = inet_addr(SERVER_IP);
    server_info.sin_port = htons(SERVER_PORT);
    server_info.sin_family = AF_INET;

    // 绑定
    if (bind(sfd, (struct sockaddr *)&server_info, sizeof(server_info)) < 0)
    {
        ERR_MSG("bind");
        return -1;
    }

    // 监听
    if (listen(sfd, 128) < 0)
    {
        ERR_MSG("listen");
        return -1;
    }

    // 客户端连接过来的信息
    struct sockaddr_in client_info;
    socklen_t len = sizeof(client_info);

    // 用于接收客户端连接的文件描述符
    int new_sfd;
    // 子进程的 pid
    pid_t cpid = -1;
    while (1)
    {
        // 连接accept
        new_sfd = accept(sfd, (struct sockaddr *)&client_info, &len);
        if (new_sfd < 0)
        {
            ERR_MSG("accept");
            return -1;
        }
        printf("[%s : %d]已连接 new_fd = %d \n", inet_ntoa(client_info.sin_addr),
               ntohs(client_info.sin_port), new_sfd);
        // 创建子进程,用于处理接收的信息
        cpid = fork();

        // 子进程进行处理
        if (0 == cpid)
        {
            // 关闭父进程的服务器文件描述符
            close(sfd);
            // 调用函数进行处理消息
            deal_client_message(new_sfd, client_info);
            // 关闭新创建的用于接收客户端传来信息的文件描述符
            close(new_sfd);
            // 结束进程,当前进程为子进程
            exit(0);
        }
        else if (cpid < 0)
        {
            ERR_MSG("fork");
            return -1;
        }

        // 下边这部分是父进程的内容
        // 在此次循环中关闭新建的文件描述符,因为下次循环又会创建了
        close(new_sfd);
    }

    // 关闭套接字
    close(sfd);
    return 0;
}

// 回调函数回收僵尸进程函数功能实现
void callBack_zembie(int sig)
{
    // 回收僵尸进程资源
    while (waitpid(-1, NULL, WNOHANG) > 0)
        ;
}

// 子进程处理客户端信息函数功能实现
int deal_client_message(int new_sfd, struct sockaddr_in client_info)
{
    char buff[128];
    ssize_t res = -1;

    while (1)
    {
        bzero(buff, sizeof(buff));
        // 接收消息,以阻塞方式接收
        res = recv(new_sfd, buff, sizeof(buff), 0);
        if (res < 0)
        {
            ERR_MSG("recv");
            break;
        }
        else if (0 == res)
        {
            // 客户端掉线
            printf("[%s : %d] 已掉线\n", inet_ntoa(client_info.sin_addr),
                   ntohs(client_info.sin_port));
            break;
        }
        // 将消息输出
        printf("[%s : %d][message : %s]\n", inet_ntoa(client_info.sin_addr),
               ntohs(client_info.sin_port), buff);

        // 判断客户端是否断开链接
        if (!strcmp(buff, "exit"))
        {
            printf("[%s : %d] 断开链接\n", inet_ntoa(client_info.sin_addr),
                   ntohs(client_info.sin_port));
            break;
        }

        strcat(buff, "---");
        // 处理消息(可以是回复)
        if (send(new_sfd, buff, sizeof(buff), 0) < 0)
        {
            ERR_MSG("send");
            break;
        }
    }
    // 关闭文件描述符
    close(new_sfd);
    return 0;
}

二、 TCP并发服务器的多线程实现

c 复制代码
#include <my_head.h>

#define SERVER_IP "192.168.125.11"      //  服务器IP
#define SERVER_PORT 6666                //  服务器端口

struct client_information
{
    int new_sfd;
    struct sockaddr_in info;
};

// 线程处理客户端信息函数
void *thread1(void *arg);

int main(int argc, const char *argv[])
{
    // 创建流式套接字
    int sfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sfd < 0)
    {
        ERR_MSG("sfd");
        return -1;
    }

    // 服务器信息
    struct sockaddr_in server_info;
    server_info.sin_addr.s_addr = inet_addr(SERVER_IP);
    server_info.sin_port = htons(SERVER_PORT);
    server_info.sin_family = AF_INET;

    // 绑定
    if (bind(sfd, (struct sockaddr *)&server_info, sizeof(server_info)) < 0)
    {
        ERR_MSG("bind");
        return -1;
    }

    // 监听
    if (listen(sfd, 128) < 0)
    {
        ERR_MSG("listen");
        return -1;
    }

    // 客户端信息
    struct client_information client_info;
    socklen_t len = sizeof(client_info.info);
    // 子线程号
    pthread_t tid;
    while (1)
    {
        // 连接accept
        client_info.new_sfd = accept(sfd, (struct sockaddr *)&(client_info.info), &len);
        if (client_info.new_sfd < 0)
        {
            ERR_MSG("accept");
            return -1;
        }
        printf("[%s : %d]已连接 new_fd = %d \n", inet_ntoa(client_info.info.sin_addr),
               ntohs(client_info.info.sin_port), client_info.new_sfd);

        // 创建子线程
        if (pthread_create(&tid, NULL, thread1, &client_info) < 0)
        {
            fprintf(stderr, "线程创建出错  __%d__\n", __LINE__);
            return -1;
        }
        // 将线程分离,待线程结束后可以直接被操作系统回收
        pthread_detach(tid);
    }

    // 关闭套接字
    close(client_info.new_sfd);
    close(sfd);
    return 0;
}

// 线程处理客户端信息函数功能实现
void *thread1(void *arg)
{
    struct client_information client_info = *((struct client_information *)arg);
    ssize_t res = -1;
    char buff[128];

    while (1)
    {
        bzero(buff, sizeof(buff));
        // 接收客户端传来的信息
        res = recv(client_info.new_sfd, buff, sizeof(buff), 0);
        if (res < 0)
        {
            ERR_MSG("recv");
            break;
        }
        else if (0 == res)
        {
            // 客户端掉线
            printf("[%s : %d] 已掉线\n", inet_ntoa(client_info.info.sin_addr),
                   ntohs(client_info.info.sin_port));
            break;
        }

        /*
         * 处理回复 */
        // 将消息输出
        printf("[%s : %d][message : %s]\n", inet_ntoa(client_info.info.sin_addr),
               ntohs(client_info.info.sin_port), buff);

        // 判断客户端是否退出
        if (!strcmp(buff, "exit"))
        {
            printf("[%s : %d] 断开链接\n", inet_ntoa(client_info.info.sin_addr),
                   ntohs(client_info.info.sin_port));
            break;
        }

        strcat(buff, "---");
        // 处理消息(可以是回复)
        if (send(client_info.new_sfd, buff, sizeof(buff), 0) < 0)
        {
            ERR_MSG("send");
            break;
        }
    }
    // 关闭文件描述符
    close(client_info.new_sfd);
    // 退出线程
    pthread_exit(NULL);
}
相关推荐
知识分享小能手11 分钟前
CentOS Stream 9入门学习教程,从入门到精通, CentOS Stream 9 命令行基础 —语法知识点与实战详解(4)
linux·学习·centos
牛奶咖啡1329 分钟前
解决配置虚拟网络后同网段的设备网络不通问题
网络·桥接模式·主机模式·配置虚拟网络后同网段设备不通·排查解决同网段同网关网络不通·重置windows主机网络·nas模式
Bona Sun1 小时前
单片机手搓掌上游戏机(二十)—pico运行doom之编译环境
c语言·c++·单片机·游戏机
车载测试工程师1 小时前
CAPL学习-ETH功能函数-通用函数
网络·学习·tcp/ip·capl·canoe
ICT技术最前线1 小时前
sdwan组网软件如何帮助企业提升网络效率?
网络·sdwan·宽带组网
烤鱼骑不快1 小时前
linux 软Raid
linux
Albert Edison1 小时前
【项目设计】C++ 高并发内存池
数据结构·c++·单例模式·哈希算法·高并发
hhwyqwqhhwy2 小时前
Linux file->private
linux·运维·服务器
WongKyunban2 小时前
在Linux下制作软件安装包
linux·运维·服务器
我真不会起名字啊2 小时前
C、C++中的sprintf和stringstream的使用
java·c语言·c++