Linux Network Thread Model

Linux Network Thread Model

  • 主线程监听请求;
  • 当有请求连接到达时,生成一个子线程处理请求的连接;
  • 子线程连接处理完成时,关闭连接,线程退出;

流程图如下:
线程模型

代码如下:

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <malloc.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <pthread.h>
#include <syslog.h>

#define PORT 80
#define BUFFER_LEN 1UL << 10

typedef struct connection{
    unsigned int fd;
    struct sockaddr_in addr;     
}Connection;

void *request_handler(void *);

int main(void)
{
    struct sockaddr_in server;
    memset(&server, 0, sizeof(struct sockaddr_in));
    
    int listenfd = socket(AF_INET, SOCK_STREAM, 0);
    
    server.sin_family = AF_INET;
    server.sin_port = htons(PORT);
    server.sin_addr.s_addr = htonl(INADDR_ANY);
    
    bind(listenfd, (struct sockaddr *)&server, sizeof(struct sockaddr_in));
    
    listen(listenfd, 255);
    
    syslog(LOG_INFO, "Thread[%ud] is listening on port[%d]......\n", pthread_self(), PORT);
    
    for(;;){
        struct sockaddr_in from;
        socklen_t len = sizeof(struct sockaddr_in);
        memset(&from, 0, sizeof(struct sockaddr_in));
        int connectfd = accept(listenfd, (struct sockaddr *)&from, &len);
        if(connectfd > 0){
            Connection* conn = (Connection *)malloc(sizeof(Connection));
            conn->fd = connectfd;
            conn->addr = from;
            pthread_t *p = (pthread_t *)malloc(sizeof(pthread_t));
            pthread_create(p, NULL, request_handler, conn);
        }
    }
    close(listenfd);
    return 0;
}


void *request_handler(void *args){
    pthread_detach(pthread_self());
    Connection *conn = (Connection *)args;
    syslog(LOG_INFO, "Thread[%ud] is handling request......\n", pthread_self());
    int connectfd = conn->fd;
    struct sockaddr_in from = conn->addr;
    char str[INET_ADDRSTRLEN];
    syslog(LOG_INFO, "receive request from %s %d\n", inet_ntop(AF_INET, &from.sin_addr, str, INET_ADDRSTRLEN), ntohs(from.sin_port));

    char buf[BUFFER_LEN];
    memset(buf, 0, BUFFER_LEN);
    read(connectfd, buf, BUFFER_LEN);
    syslog(LOG_INFO, "The message is : %s\n", buf);
    char* response = "I got your message!\n";
    write(connectfd, response, strlen(response));
    close(connectfd);
    free(conn);
    pthread_exit(NULL);
}

备注说明:

  • 在子线程处理函数void *request_handler(void *args)中,需要将子线程与主线程进行分离操作,即pthread_detach(pthread_self());否则的话,子线程结束后资源不会被系统自动回收,从而导致thread memory leak;
  • 子线程退出之前,需要关闭connectfd,否则会导致应用程序打开过多的套接字而导致error=24, too many open files.
当出现"Network is unreachable"的错误消息时,这意味着您的Linux系统无法连接到网络。这可能是由于以下几个原因导致的: 1. 网络配置错误:请检查您的网络配置是否正确,包括IP地址、子网掩码、默认网关和DNS服务器设置。 2. 网络服务未启动:请确保网络服务已经启动。在Linux 6上,您可以使用命令"service network restart"来重启网络服务。在Linux 7上,可以使用命令"systemctl restart network.service"来重启网络服务。 3. 网络硬件故障:请检查您的网络硬件设备,例如网卡、路由器等是否正常工作。 4. 防火墙设置:防火墙可能阻止您的Linux系统与外部网络进行通信。请检查防火墙设置,并确保允许所需的网络流量通过。 解决此问题的步骤如下: 1. 检查网络配置是否正确,并确保IP地址、子网掩码、默认网关和DNS服务器设置正确。 2. 检查网络服务是否已启动。在Linux 6上,您可以使用命令"service network restart"来重启网络服务。在Linux 7上,可以使用命令"systemctl restart network.service"来重启网络服务。 3. 检查网络硬件设备是否正常工作,例如网卡是否正确插入、电缆是否连接好等。 4. 检查防火墙设置,并确保允许所需的网络流量通过。您可以使用iptables命令来配置防火墙规则。 5. 如果以上步骤都没有解决问题,请联系网络管理员或您的网络服务提供商以获取更多帮助。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值