热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

使用进程池实现高并发服务器

近期在将《Linux高性能服务器编程》的代码整理出一个实验板服务器demo,陆陆续续将知识点梳理出来,本文主要实现进程池。进程池的必要性为什么要使用进程池线程池?cause:1、动

近期在将《Linux高性能服务器编程》的代码整理出一个实验板服务器demo,陆陆续续将知识点梳理出来,本文主要实现进程池。

进程池的必要性

为什么要使用进程池/线程池?cause:

1、动态创建进程/线程比较耗费时间,会导致比较慢的客户响应;

2、动态申请的进程/线程通常只用来为一条连接服务,这会导致系统产生大量的进程/线程,进程切换会消耗大量CPU时间;

3、动态创建的子进程是当前进程的完整镜像。因此当前进程必须谨慎管理其分配的fd、socket和堆内存等系统资源,否则子进程复制这些资源后会导致系统可用资源下降,从而影响服务器性能。

进程池实现的并发服务器可以解决上述问题。进程池是由服务器预先创建的一组子进程,所有子进程都运行相同的代码,并且具有相同的属性,比如优先级、PGID(进程组)等,没有打开不必要的文件描述符,也不会使用大块的堆内存。当有新的客户端连接到来时,主进程通过均衡算法从进程池选择某一子进程来服务,其代价比临时创建进程小得多。至于如何选择子进程,有两种方式:

1、随机/轮流选择空闲子进程

2、主进程和所有子进程共享一个工作队列,子进程都睡眠在该队列上,当有新连接到来时唤醒某一个子进程。

选择好子进程后,主进程还需要通过某种机制通知目标子进程有新任务需要处理,并传递必要的数据。最简单的方法是父子进程预先建立好管道。对于线程池而言就不需要管道了,因为父子线程本身就是共享全局数据的。

综上进程池的实现方式为:

                         

使用进程池实现高并发服务器

 

处理多客户

使用进程池处理并发连接需要考虑的一个问题是:listen socket和connect socket事件是否都由主进程管理。

服务器程序通常需要处理三类事件:IO事件、信号signal、定时事件,同步IO模型通常用于实现reactor模式,异步IO模型用于实现proactor模式。结合同步异步IO模型,可以设计出下面两种服务器事件处理模式:

使用进程池实现高并发服务器

半同步/半反应堆模式

使用进程池实现高并发服务器

半同步/半异步模式

IO复用模型的具体介绍参考UNIX网络编程第六章,这里不展开。

使用进程池实现高并发服务器

半同步/半异步模式

第一种模式由主进程统一管理两种socket,因为主从进程共享请求队列,进程对队列的任何操作都需要加锁,影响CPU性能。第二种模式主进程管理所有监听socket,子进程分别管理自己的连接socket。子进程可以自己调用accept获取TCP连接队列,而主进程只需要通知就行,具体通知实现方式:

在两个进程之间建立一个Unix域socket作为消息传递的通道(使用 socketpair 函数),然后父进程调用 sendmsg 向通道发送一个特殊的消息,内核将对这个消息做特殊处理,从而将消息传递到子进程,子进程调用recvmsg 从通道接收消息。

此外,在设计进程池时还需要考虑同一个客户端的多次请求是否可以复用一个TCP连接(http层面添加keepalive字段即可),如果客户请求是无状态的,那么服务器可以实现使用不同子进程来为该客户的不同请求进行服务:

使用进程池实现高并发服务器

如果客户任务是有上下文状态的(比如购物车,总不能刷新一次就被清空了),那么只能一致用同一个子进程来处理该客户的请求。使用epoll的EPOLLONESHOT特性可以确保客户连接在整个生命周期仅被一个进程处理。

对于注册了EPOLLONESHOT事件的socket,操作系统最多触发一次其注册的事件,且只触发一次。因此当子进程处理该socket上的事件时,其他进程不可能操作该socket。

具体实现

定义一个描述子进程的类process,包含子进程的PID,以及父子进程通信的管道pipe;进程池使用单体模式,保证程序仅创建一个进程池实例,这是程序正确处理信号的必要条件:

使用进程池实现高并发服务器

下面是父进程启动后创建的数据,包括:处理信号的管道、承载子进程信息的数组、进程标识和epoll事件表,并进行了初始化。因此调用fork生成子进程后会继承这些数据,子进程会将自己的数据保存。

使用进程池实现高并发服务器

父进程的执行流程为:

使用进程池实现高并发服务器

关于SIGNAL信号的更多知识,参考UNIX环境高级编程第十章:

使用进程池实现高并发服务器

进程池中父子进程使用信号实现通信的实现如下:

使用进程池实现高并发服务器

子进程实现

子进程的数据如下:

使用进程池实现高并发服务器

子进程处理流程如下:

使用进程池实现高并发服务器

在使用进程池时,监听socket一般由main函数创建,在退出后需要关闭该socket。下期讲解如何使用该进程池实现一个简单的CGI服务器。


推荐阅读
  • 本文介绍了操作系统的定义和功能,包括操作系统的本质、用户界面以及系统调用的分类。同时还介绍了进程和线程的区别,包括进程和线程的定义和作用。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • 本文介绍了Python高级网络编程及TCP/IP协议簇的OSI七层模型。首先简单介绍了七层模型的各层及其封装解封装过程。然后讨论了程序开发中涉及到的网络通信内容,主要包括TCP协议、UDP协议和IPV4协议。最后还介绍了socket编程、聊天socket实现、远程执行命令、上传文件、socketserver及其源码分析等相关内容。 ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • 本文讨论了clone的fork与pthread_create创建线程的不同之处。进程是一个指令执行流及其执行环境,其执行环境是一个系统资源的集合。在调用系统调用fork创建一个进程时,子进程只是完全复制父进程的资源,这样得到的子进程独立于父进程,具有良好的并发性。但是二者之间的通讯需要通过专门的通讯机制,另外通过fork创建子进程系统开销很大。因此,在某些情况下,使用clone或pthread_create创建线程可能更加高效。 ... [详细]
  • 【重识云原生】第四章云网络4.8.3.2节——Open vSwitch工作原理详解
    2OpenvSwitch架构2.1OVS整体架构ovs-vswitchd:守护程序,实现交换功能,和Linux内核兼容模块一起,实现基于流的交换flow-basedswitchin ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • 本文介绍了使用PHP实现断点续传乱序合并文件的方法和源码。由于网络原因,文件需要分割成多个部分发送,因此无法按顺序接收。文章中提供了merge2.php的源码,通过使用shuffle函数打乱文件读取顺序,实现了乱序合并文件的功能。同时,还介绍了filesize、glob、unlink、fopen等相关函数的使用。阅读本文可以了解如何使用PHP实现断点续传乱序合并文件的具体步骤。 ... [详细]
  • 深入理解Kafka服务端请求队列中请求的处理
    本文深入分析了Kafka服务端请求队列中请求的处理过程,详细介绍了请求的封装和放入请求队列的过程,以及处理请求的线程池的创建和容量设置。通过场景分析、图示说明和源码分析,帮助读者更好地理解Kafka服务端的工作原理。 ... [详细]
  • Android工程师面试准备及设计模式使用场景
    本文介绍了Android工程师面试准备的经验,包括面试流程和重点准备内容。同时,还介绍了建造者模式的使用场景,以及在Android开发中的具体应用。 ... [详细]
  • STL迭代器的种类及其功能介绍
    本文介绍了标准模板库(STL)定义的五种迭代器的种类和功能。通过图表展示了这几种迭代器之间的关系,并详细描述了各个迭代器的功能和使用方法。其中,输入迭代器用于从容器中读取元素,输出迭代器用于向容器中写入元素,正向迭代器是输入迭代器和输出迭代器的组合。本文的目的是帮助读者更好地理解STL迭代器的使用方法和特点。 ... [详细]
  • 深入解析Linux下的I/O多路转接epoll技术
    本文深入解析了Linux下的I/O多路转接epoll技术,介绍了select和poll函数的问题,以及epoll函数的设计和优点。同时讲解了epoll函数的使用方法,包括epoll_create和epoll_ctl两个系统调用。 ... [详细]
  • linux进阶50——无锁CAS
    1.概念比较并交换(compareandswap,CAS),是原⼦操作的⼀种,可⽤于在多线程编程中实现不被打断的数据交换操作࿰ ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 006_Redis的List数据类型
    1.List类型是一个链表结构的集合,主要功能有push,pop,获取元素等。List类型是一个双端链表的结构,我们可以通过相关操作进行集合的头部或者尾部添加删除元素,List的设 ... [详细]
author-avatar
yanna00799
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有