PHP的多进程编程pcntl

阿里云服务器3折开售(点此直达)

image

背景:
最近写项目的时候,遇到数据库需要加行独占锁for update的事情。然后,突然想看看,加锁和不加锁的性能问题,看看到底会慢下来多少。

可是PHP这个渣渣,在apache或者nginx中容器内根本用不了多进程或者多线程之类的搞并发啊测试一波啊。
吐槽一下:木有Java实在,用java开多线程,那叫一个爽,代码方便优雅。没办法,谁让现在使用的是PHP呢,只能cli模式了。

又不能用线上测,平时使用的测试环境也用不了。因为PHP是NTS的。所以,又在自己本地的测试机器上面搭了一套ZTS版本的。

然后,发现所有的精力都放在鼓捣这个玩意儿上面了。PCNTL这个扩展挺好玩。哈哈……把我深深的吸引住了。

不扯犊子了,开始开始……

PHP的多进程 依赖 pcntl 插件
所以,要么就重新编译PHP,然后 php -m | grep pcntl 看看有没有装成功
要么就从源码目录 php-src/ext/pcntl 下 phpize 来生成 configure 文件,
./configure .... make
命令:

//xxx 代表着 自己的php版本的安装目录
xxx/phpxxx/bin/phpize --with-php-config=xxx/phpxxx/bin/php-config 

然后,在pcntl的modules目录下将 so文件 导入到自己的 php扩展目录下,还得在 php.ini 文件中,增加这个扩展。

如果有条件的话,还是再自行编译一版php吧。./configure --enable-pcntl 这样的话,可以将pcntl 打包到 php 的内部

还得有 posix 扩展,这个貌似是默认自己就装上了。

注意事项:
1、使用自己版本的php的phpize,不然一会儿编译好或者make install的时候,so文件就给你安装的别的地方去了。因为我的机器上面安装了多个版本的php环境,而测试用的是其中一个版本。所以需要指定使用哪个版本的 phpize
2、xxx/phpxxx/bin/phpize --with-php-config=xxx/phpxxx/bin/php-config 运行配置,如果你的服务器上只是装了一个版本的php则不需要添加--with-php-config 。后面的参数只是为了告诉phpize要建立基于哪个版本的扩展。make install 让它自己安装上,也行。

下面是代码:
简单的 fork

$pid = pcntl_fork();
if($pid>0){
        echo 'father'.PHP_EOL;
        echo ' father pid'.posix_getpid().PHP_EOL;
        echo 'father group id'.posix_getgid().PHP_EOL;
}else{
        echo 'son'.PHP_EOL;
        echo ' son pid'.posix_getpid().PHP_EOL;
        echo 'son group id'.posix_getgid().PHP_EOL;
}

结果:

father
 father pid9831
father group id1013
son
 son pid9832
son group id1013

可以看到, fork 之后,主子进程都会从 fork 处,继续向下执行。
1013 是当前运行账户的 组ID

本来这个事情这样看着还比较简单,但是这个事儿,一旦加上循环。。。可就不简单了。
来,上一段代码:

for($i=0;$i<3;$i++){
    echo 'this loop index= '.$i.PHP_EOL;
    $pid = pcntl_fork();
    if($pid > 0){
        echo ' i was the father,my pid is '.posix_getpid().PHP_EOL;
        echo '111'.PHP_EOL;
    }elseif($pid == 0){
        echo ' i was child process,my pid is '.posix_getpid().PHP_EOL;
        echo '222'.PHP_EOL;
        exit();
    }else{
            echo 'fork failed'.PHP_EOL;
            echo '333'.PHP_EOL;
    }
}

结果

this loop index= 0
 i was the father,my pid is 10639
111   // 这是主进程的输出,然后第一次循环就结束了
this loop index= 1   //这时候主进程不能停下来啊,所以进行第二次循环,输出这句话
 i was child process,my pid is 10640 //这时候为啥是子进程开始输出了呢?说明主进程遇到了 fork 函数呗,这时候将cpu 切换给子进程运行一下子,所以就有了这句话呗
222    // 子进程得到了cpu不能轻易撒手啊,得继续往下走啊,所有有了这句话
 i was the father,my pid is 10639 //这时候为啥又切回主进程了呢?因为啊,上面的第一次循环的子进程遇到了exit 之后死了呗。然后,cpu就切回来了
111   // 主进程继续干活儿
this loop index= 2 // 主进程干完输出 111 的活儿后,该进行第三次循环了啊。所以,输出这里
 i was the father,my pid is 10639 // 这是第三次循环时的,主进程输出
111 // 主进程继续往下执行。主进线进行到这里了,for循环该结束了啊。可它不能结束啊,因为,它的子进程还有任务呢,所以,它就得在这儿等会儿。不然,父进行死了,那子进程不就成了僵尸进程了麽
 i was child process,my pid is 10642 // 所以,这个是第三次循环的子进程,它先执行了(先的意思是,对比第二次循环生成的子进程。这个我估计不一定谁先执行,这个玩意儿没准儿) 
222 // 然后,输出第三次循环子进程的内容
 i was child process,my pid is 10641 // 这时候,没人跟你抢CPU的资源了啊,所以,将第二次循环的子进程运行一下子
222 // 第二次循环的子进程内容输出

上面解释的话,看看结果中的注释就差不多了。
如果不够明显的话,再来个例子:将每次输出的 内容,都加上循环的 标记。这样就可以知道,进程是如何切换的了

for($i=0;$i<3;$i++){
    echo 'this loop index= '.$i.PHP_EOL;
    $pid = pcntl_fork();
    if($pid > 0){
        echo ' i was the father,my pid is '.posix_getpid().PHP_EOL;
        echo 'num='.$i.',111'.PHP_EOL;
    }elseif($pid == 0){
        echo ' i was child process,my pid is '.posix_getpid().PHP_EOL;
        echo 'num='.$i.',222'.PHP_EOL;
        exit();
    }else{
        echo 'fork failed'.PHP_EOL;
        echo 'num='.$i.',333'.PHP_EOL;
    }
}

结果

this loop index= 0
 i was the father,my pid is 10794
num=0,111
this loop index= 1
 i was child process,my pid is 10795
 i was the father,my pid is 10794
num=0,222
num=1,111
this loop index= 2
 i was the father,my pid is 10794
num=2,111
 i was child process,my pid is 10796
 i was child process,my pid is 10797
num=1,222
num=2,222

所以说,每次运行的结果都是不一样的。

上面这个还是比较正常的,但是,一旦把 子进程中的 exit 函数去掉之后,喔哦~~~哦呦。。。了不得了。。。
来来来瞅瞅
代码:

for($i=0;$i<3;$i++){
    echo 'this loop index= '.$i.PHP_EOL;
    $pid = pcntl_fork();
    if($pid > 0){
        echo ' i was the father,my pid is '.posix_getpid().PHP_EOL;
        echo '111'.PHP_EOL;
    }elseif($pid == 0){
        echo ' i was child process,my pid is '.posix_getpid().PHP_EOL;
        echo '222'.PHP_EOL;
       // exit(); //把我注释掉,这次热闹了
    }else{
            echo 'fork failed'.PHP_EOL;
            echo '333'.PHP_EOL;
    }
}

结果:

this loop index= 0
 i was the father,my pid is 10665
111
this loop index= 1
 i was child process,my pid is 10666
222
this loop index= 1
 i was the father,my pid is 10665
111
this loop index= 2
 i was the father,my pid is 10666
111
this loop index= 2
 i was the father,my pid is 10665
111
 i was child process,my pid is 10668
222
this loop index= 2
 i was the father,my pid is 10666
111
 i was the father,my pid is 10668
111
root@bakup:~/pcntl_test#  i was child process,my pid is 10667
222
this loop index= 2
 i was child process,my pid is 10670
 i was the father,my pid is 10667
111
 i was child process,my pid is 10672
222
222
 i was child process,my pid is 10669
 i was child process,my pid is 10671
222
222

这个理解起来,是 迷之乱啊。。。
但是中心思想就是

第一次循环的子进程内会继续从$i=0 开始 执行到 3。所以,执行3次
第二次循环,子进程内,会继续从 $i=1,开始执行到3,所以执行2次
第三次循环,子进程内,会继续从 $i=2 开始执行到3,所以执行1次。

所以,最终看到的结果,子进程 一共执行了 7 次。(即有 7个 ‘222’)

有时间再慢慢给结果加注释吧。

看了这么多网上的文章,没有一个人能把这个事儿说清楚的。
我觉得,为了更直观,可以采用 上面的方法,给 ‘222‘或者pid的语句前面,加上循环的 标记序号,这样就知道,这是第几次循环的内容了。(但一定得加到一个 echo 里面,是一个整体,防止 cpu切走。)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,293评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,604评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,958评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,729评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,719评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,630评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,000评论 3 397
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,665评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,909评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,646评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,726评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,400评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,986评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,959评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,996评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,481评论 2 342