背景:
最近写项目的时候,遇到数据库需要加行独占锁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切走。)