PHPSocket编程之多进程的回声服务器

左手的ㄟ右手 2024-04-03 08:32 184阅读 0赞

采用多进程的方式来实现同时为多个客户端提供服务。

以下是最终的效果:

8c8188aa0b5521b1af6a31a2ef6bff73.png

  1. (通过 ps 命令可以查看到我们创建的工作进程)

1ab2aa73914d80c1b41b3321755e78a9.png

  1. (通过 xshell 打开多个终端,并向服务程序连接多个客户端)

在开始编码之前,先介绍一下什么是进程。

进程,即正在运行的占用内存的程序,是一个独立的内存空间。比如我们在 windows 上打开记事本软件,这个操作就相当于打开了一个进程。

而多进程则是通过创建多个进程来共同完成一件事。

我们本次只需实现服务端的程序就行了,客户端还使用之前的。没读过之前文章的可以查看历史文章。

在PHP中,我们可以使用 pcntl_fork 函数来实现创建进程。

下面是函数的原型:

int pcntl_fork ( void )

官方解释:

成功时,在父进程执行线程内返回产生的子进程的PID,在子进程执行线程内返回0。失败时,在 父进程上下文返回-1,不会创建子进程,并且会引发一个PHP错误。

为什么说是父进程执行的线程呢,因为一个进程中至少包含一个线程 ,而这个线程则是进程的主线程。

父进程在调用 pcntl_fork 时,同时复制出一个独立的子进程,这个子进程具有父进程同样的上下文。也就是说两个进程共享一个代码而已。

下面进入编码环节:

466590ef9c3b4a5096793f51a77ea1fe.png

通过执行 pcntl_fork 函数,同时复制了一个子进程,此时,如果上下文是父进程的执行环境,则返回值为子进程的进程号。如果是子进程的执行环境,则返回0。所以下面的if程序结构,两个分支都执行了。

子进程同样还是以前的逻辑,用来接收客户端的消息,同时发送给客户端。

倒数第二行调用了 pcntl_waitpid 函数,在讲解这个函数之前我们先了解一下什么是僵尸进程。

正常情况下,子进程是通过父进程创建的。由于进程是互相独立的内存结构,所以父进程是不会知道子进程的运行状态的。子进程完成自己的任务之后,并不能自己退出,这个时候需要父进程通过操作系统来取得子进程的状态,从而回收子进程。否则,我们的子进程将会成为一个垃圾资源,也就是僵尸进程。

下面是这个函数的函数原型:

int pcntl_waitpid ( int $pid , int &$status [, int $options = 0 ] )

官方解释:

等待或返回fork的子进程状态。

其实上面的代码是有问题的,在 foreach 中第一次调用 pcntl_waitpid 之后 ,主进程其实此时是被阻塞着,一直在等待第一个子进程退出,而其他的子进程若此时异常退出,则并没有被主进程回收,也就产生了僵尸进程。而在实际的开发中主进程还是要做其他的事情的。

所以这里推荐采用非阻塞的方式,很简单,只需加上第三个参数:WNOHANG。即:pcntl_waitpid ($pid, $status, WNOHANG)

这样子可以在没有子进程退出的情况下立刻返回,从而继续执行后续代码。

发表评论

表情:
评论列表 (有 0 条评论,184人围观)

还没有评论,来说两句吧...

相关阅读

    相关 进程并发服务器编程

    多进程并发服务器编程 一、实验目的 理解进程的创建和终止方法; 熟悉父进程与子进程对描述符的操作过程; 学会编写基本的多进程并发服务器程序和客户程序。 二、实验平

    相关 C++回声服务器_3-UDP版本

    这次我们实现一个UDP版本的回声服务器。 用于传输数据的函数 UDP套接字不会像TCP套接字那样保持连接状态,因此每次传输数据都要添加目标地址信息。 用于传输数据的函

    相关 进程编程

    多进程编程 概念 程序: 计算机中存储的二进制代码 进程: 计算机中运行的程序 多进程: 多个进程同时运行 进程的创建 Linux环境下执行 实例