Linux网络编程(高并发服务器)

2023-08-17 08:54:39


前言

本篇文章带大家学习Linux网络编程中的高并发服务器。首先我们需要了解什么是高并发服务器,然后是学习如何来编写高并发服务器。

一、什么是高并发服务器

高并发服务器是指能够同时处理大量并发请求的服务器系统。在网络应用中,当多个用户或客户端同时请求服务器时,服务器需要能够高效地处理这些请求,并且保持良好的性能和稳定性。

高并发服务器的设计和实现需要考虑以下几个关键因素:

1.多线程或多进程处理:采用多线程或多进程的方式可以使服务器能够同时处理多个请求。每个线程或进程负责处理一个请求,从而提高服务器的并发处理能力。

2.异步非阻塞I/O:使用异步非阻塞I/O编程模型可以避免在请求处理过程中阻塞线程或进程,充分利用系统资源,提高服务器的并发处理性能。常见的方式是使用事件驱动的编程框架或库,例如Node.js的Event-driven I/O、Nginx的事件驱动模型等。

3.负载均衡:通过在服务器集群中引入负载均衡器,将请求分发到多个服务器节点上,可以进一步提高整个系统的并发处理能力。负载均衡器可以根据一定的策略(如轮询、权重等)将请求分发到不同的服务器,使得每个服务器都能够处理适量的请求。

4.缓存和分布式存储:合理地使用缓存和分布式存储可以降低服务器的负载并提高响应速度。将频繁访问的数据存储在缓存中,减轻对后端存储系统的压力。

5.水平扩展:通过增加服务器的数量来扩展系统的处理能力,例如增加服务器节点或使用云计算服务提供商的弹性伸缩功能。水平扩展可以使系统能够处理更多的并发请求。

二、使用多线程和多进程实现高并发服务器的思路

TCP通信中其实客户端连接上服务器后,服务器会新创建一个客户端出来与连接的客户端进行通信,而不是直接进行通信。这样的话我们就有思路了,我们需要让服务器一直处于accpet等待连接的状态,等待新的客户端连接上来,当有客户端连接上来后会创建一个新的客户端与之进行通信,那么我们就需要为这个客户端去创建一个线程或者是进程,这样的话就不会影响到服务器的接收新客户端的连接请求了。

在这里插入图片描述

三、多进程服务器代码编写

客户端连接服务器成功后就使用fork函数创建出子进程和客户端进行通信,父进程使用信号来对子进程进行回收。

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <signal.h>
#include <sys/wait.h>
#include <errno.h>

/*回收子进程*/
void catch_child(int sig) 
{
    pid_t pid;
    int status;
    while ((pid = waitpid(-1, &status, WNOHANG)) > 0) 
    {
        // 处理子进程的退出状态
    }
}

int main()
{
    int server = 0;
    struct sockaddr_in saddr = {0};
    int client = 0;
    struct sockaddr_in caddr = {0};
    socklen_t asize = 0;
    int len = 0;
    char buf[32] = {0};
    int r = 0;

    pid_t pid;

    server = socket(PF_INET, SOCK_STREAM, 0);

    if( server == -1 )
    {
        printf("server socket error\n");
        return -1;
    }

    saddr.sin_family = AF_INET;
    saddr.sin_addr.s_addr = htonl(INADDR_ANY);
    saddr.sin_port = htons(8888);

    if( bind(server, (struct sockaddr*)&saddr, sizeof(saddr)) == -1 )
    {
        printf("server bind error\n");
        return -1;
    }

    if( listen(server, 128) == -1 )
    {
        printf("server listen error\n");
        return -1;
    }

    printf("server start success\n");

    while( 1 )
    {
        asize = sizeof(caddr);      
        client = accept(server, (struct sockaddr*)&caddr, &asize);

        if( client == -1 )
        {
            if (errno == EINTR)
            {
                // 信号中断,重新调用accept
                client = accept(server, (struct sockaddr*)&caddr, &asize);
            }
            else
            {
                perror("accept");
                printf("client accept error\n");
                return -1;
            }
        }   

        pid = fork();//创建子进程与客户端进行通信

        if(pid == 0)
        {
            close(server);
            while (1)
            {
                /*子进程*/
                len = read(client, buf, 1024);
                if(len == 0)
                {
                    printf("child exit\n");
                    close(client);
                    exit(1);//退出子进程
                }
                write(client, buf, len);
                printf("recv_buf : %s len : %d\n", buf, len);
                printf("child pid : %d\n", getpid());
            }                       
        }
        else if(pid > 0)
        {
            /*父进程*/
            struct sigaction act;
            sigemptyset(&act.sa_mask);
            act.sa_handler = catch_child;            
            act.sa_flags = 0;

            sigaction(SIGCHLD, &act, NULL);

            close(client);                    
        }
    }
    
    close(server);

    return 0;
}

四、多线程服务器代码编写

使用多线程的方法比多进程的方法简单一些,这里使用pthread_detach函数将线程进行分离,这样的话就不需要我们手动的去回收线程了,我们要做的就是在线程函数里面和连接上来的客户端进行通信就可以了。

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <pthread.h>

void* client_work(void* arg)
{
    int client = (int)arg;
    int len = 0;
    char buf[1024];

    while (1)
    {
        len = read(client, buf, 1024);
        if(len == 0)
        {
            printf("client is close\n");
            return NULL;
        }
        printf("read buf : %s\n", buf);
        write(client, buf, len);
    }    

    close(client);
    return NULL;
}

int main()
{
    int server = 0;
    struct sockaddr_in saddr = {0};
    int client = 0;
    struct sockaddr_in caddr = {0};
    socklen_t asize = 0;
    int len = 0;
    char buf[32] = {0};
    int r = 0;

    pid_t pid;

    server = socket(PF_INET, SOCK_STREAM, 0);

    if( server == -1 )
    {
        printf("server socket error\n");
        return -1;
    }

    saddr.sin_family = AF_INET;
    saddr.sin_addr.s_addr = htonl(INADDR_ANY);
    saddr.sin_port = htons(8888);

    if( bind(server, (struct sockaddr*)&saddr, sizeof(saddr)) == -1 )
    {
        printf("server bind error\n");
        return -1;
    }

    if( listen(server, 128) == -1 )
    {
        printf("server listen error\n");
        return -1;
    }

    printf("server start success\n");

    while( 1 )
    {
        pthread_t tid;
        asize = sizeof(caddr);      
        client = accept(server, (struct sockaddr*)&caddr, &asize);

        if( client == -1 )
        {
            if (errno == EINTR)
            {
                // 信号中断,重新调用accept
                client = accept(server, (struct sockaddr*)&caddr, &asize);
            }
            else
            {
                perror("accept");
                printf("client accept error\n");
                return -1;
            }
        }

        pthread_create(&tid, NULL, client_work, (void*)client);   
        pthread_detach(tid);
    }
    
    close(server);

    return 0;
}

总结

本篇文章就讲解到这里。

更多推荐

科技资讯|Canalys发布全球可穿戴腕带设备报告,智能可穿戴增长将持续

市场调查机构Canalys近日发布报告,表示2023年第2季度全球可穿戴腕带设备出货量达4400万台,同比增长了6%。主要归功于其亲民的价格以及消费者对价位较高的替代品仍持谨慎态度,基础手环市场尽管与去年同期相比有所下降,仍然保持稳定的市场份额,约为19%。可穿戴设备仍然具有长期的发展前景。尽管短期经济因素使消费者更倾

聊聊Spring中循环依赖与三级缓存

先看几个问题什么事循环依赖?什么情况下循环依赖可以被处理?spring是如何解决循环依赖的?什么是循环依赖?简单理解就是实例A依赖实例B的同时B也依赖了A@ComponentpublicclassA{//A中依赖B@AutowiredprivateBb;}@ComponentpublicclassB{//B中依赖A@A

spring boot 八、 sharding-jdbc 分库分表 按月分表

在项目resources目录下新建com.jianmu.config.sharding.DateShardingAlgorithm文件新增yaml配置数据源spring:shardingsphere:props:sql:#是否在日志中打印SQLshow:true#打印简单风格的SQLsimple:truedatasou

JavaWeb概念视频笔记

学习地址:102.尚硅谷_Tomcat-Tomcat服务器和Servlet版本的对应关系_哔哩哔哩_bilibili目录1.JavaWeb的概念2.Web资源的分类3.常用的Web服务器4.Tomcat服务器和Servlet版本的对应关系5.Tomcat的使用a.安装b.目录介绍c.如何启动Tomcat服务器另一种启动

9.19号作业

2>完成文本编辑器的保存工作widget.h#ifndefWIDGET_H#defineWIDGET_H#include<QWidget>#include<QFontDialog>#include<QFont>#include<QMessageBox>#include<QDebug>#include<QColorDia

[npm]脚手架本地全局安装1

[npm]脚手架本地全局安装1npmlink全局安装npminstall全局安装卸载全局安装的脚手架该文章是你的脚手架已经开发完成的前提下,你想要本地全局安装该脚手架,便于本地使用脚手架的命令的情况npmlink全局安装如果本地开发的项目是个脚手架,只是个人使用,也并不需要上传到npm或者私库,如何安装本地的项目到包的

设计模式(2) - 创建型模式

创建型模式指的是创建对象或是获取实例的方式。1、工厂模式平时写一些简单的代码可能会直接用new创建出一个对象,但是实际在阅读一些功能比较多、规模比较庞大的工程时,可能会发现有多个类继承于同一个基类的情况,它们拥有同样的接口但是实现了不同的功能。它们可能是可以互相替代的两套系统(例如AndroidMedia中的ACode

Docker 容器设置为自动重启

Docker自动重启原因Docker自动重启通常是由以下几个原因导致的:程序崩溃系统内存不足系统进程使用过多CPU和RAM导致的阻塞docker容器被杀死或重新启动,导致应用程序中断网络中断当这些问题出现时,Docker会自动重启运行中的服务来尝试解决问题。dockerupdate--restart=alwaysmys

【docker】容器跟宿主机、其他容器通信

说明容器跟宿主机、其他容器通信的关键在于它们要在同一个网络,或者通过修改路由信息来可以让它们互相之间能够找得到对方的IP。本文主要介绍让它们在同一个网络的方法。Docker自定义网络模式介绍Docker容器可以通过自定义网络来与宿主机或其他容器进行通信。在Docker中,有三种类型的网络:bridge网络、host网络

使用ElementPlus实现内嵌表格和内嵌分页

前言有时遇到这样的需求,就是在表格里面嵌入一个表格,以及要求带有分页,这样在ElementPlus中很好实现。以下使用Vue2语法实现一个简单例子,毕竟Vue3兼容Vue2语法,若想要Vue3版本例子,简单改改就OK了。一、示例代码(1)/src/views/Example/InlineTable/index.vue<

proteus中的各种电阻-可变电阻-排阻

在原理图中使用各类型的电阻是很常见的事情,尤其类似与排阻、可变电阻,但这些电阻对于不熟悉proteus的童鞋来说,一下子可能很难找到,或者很难找心中所想的那个类型,这里分类列出,便于大家使用。文章目录一、普通电阻1、普通电阻(模拟信号电阻)2、上拉与下拉电阻(模拟信号电阻常常报错)二、排阻1、ResistorPack2

热文推荐