色综合图-色综合图片-色综合图片二区150p-色综合图区-玖玖国产精品视频-玖玖香蕉视频

您的位置:首頁技術(shù)文章
文章詳情頁

php 的多進(jìn)程操作實踐案例分析

瀏覽:135日期:2022-09-11 13:01:58

本文實例講述了php 的多進(jìn)程操作。分享給大家供大家參考,具體如下:

php的多進(jìn)程處理依賴于pcntl擴(kuò)展,通過pcntl_fork創(chuàng)建子進(jìn)程來進(jìn)行并行處理。

例1如下:

<?php$pid = pcntl_fork();if($pid == -1) { //錯誤處理:創(chuàng)建子進(jìn)程失敗時返回-1. die(’fork error’);} else if ($pid) { //父進(jìn)程會得到子進(jìn)程號,所以這里是父進(jìn)程執(zhí)行的邏輯 echo 'parent n'; //等待子進(jìn)程中斷,防止子進(jìn)程成為僵尸進(jìn)程。 pcntl_wait($status);} else { //子進(jìn)程得到的$pid為0, 所以這里是子進(jìn)程執(zhí)行的邏輯。 echo 'child n'; exit;}

pcntl_fork創(chuàng)建了子進(jìn)程,父進(jìn)程和子進(jìn)程都繼續(xù)向下執(zhí)行,而不同是父進(jìn)程會獲取子進(jìn)程的$pid也就是$pid不為零。而子進(jìn)程會獲取$pid為零。通過if else語句判斷$pid我們就可以在指定位置寫上不同的邏輯代碼。

上述代碼會分別輸出parent和child。那么輸出的parent和child是否會有順序之分?是父進(jìn)程會先執(zhí)行?

例2如下:

<?php$pid = pcntl_fork();if($pid == -1) { die(’fork error’);} else if ($pid) { sleep(3); echo 'parent n'; pcntl_wait($status);} else { echo 'child n'; exit;}

我們在父進(jìn)程中通過sleep來延緩執(zhí)行,看看效果。

結(jié)果是,很快輸出了child,等待了接近3秒后,才輸出parent。所以父進(jìn)程和子進(jìn)程的執(zhí)行是相對獨立的,沒有先后之分。

那么問題又來了?pcntl_wait是做什么用的?會掛起當(dāng)前進(jìn)程,直到子進(jìn)程退出,如果子進(jìn)程在調(diào)用此函數(shù)之前就已退出,此函數(shù)會立刻返回。子進(jìn)程使用的資源將被釋放。

例3如下:

<?php$pid = pcntl_fork();if($pid == -1) { die(’fork error’);} else if ($pid) { pcntl_wait ($status); echo 'parent n';} else { sleep(3); echo 'child n'; exit;}

上述代碼,我們可以看到,父進(jìn)程執(zhí)行pcntl_wait時就已經(jīng)掛起,直到等待3秒后輸出child,子進(jìn)程退出后。父進(jìn)程繼續(xù)執(zhí)行,輸出parent。

例4如下:

<?phpdefine(’FORK_NUMS’, 3);$pids = array();for($i = 0; $i < FORK_NUMS; ++$i) { $pids[$i] = pcntl_fork(); if($pids[$i] == -1) { die(’fork error’); } else if ($pids[$i]) { pcntl_waitpid($pids[$i], $status); echo 'pernet n'; } else { sleep(3); echo 'child id:' . getmypid() . ' n'; exit; }}

上述代碼,我們創(chuàng)建3個子進(jìn)程,父進(jìn)程分別掛起等待子進(jìn)程結(jié)束后,輸出parent。

輸出結(jié)果如下:

child id:19090pernetchild id:19091pernetchild id:19092pernet

例5如下:

<?phpdefine(’FORK_NUMS’, 3);$pids = array();for($i = 0; $i < FORK_NUMS; ++$i) { $pids[$i] = pcntl_fork(); if($pids[$i] == -1) { die(’fork error’); } else if ($pids[$i]) { } else { sleep(3); echo 'child id:' . getmypid() . ' n'; exit; }}foreach($pids as $k => $v) { if($v) { pcntl_waitpid($v, $status); echo 'parent n'; }}

輸出結(jié)果如下:

child id:19118child id:19119child id:19120parentparentparent

為什么上述代碼跟例4的輸出結(jié)果不一樣?

我們可以看到例5的pcntl_waitpid函數(shù)放在了foreach中,foreach代碼是在主進(jìn)程中,也就是父進(jìn)程的代碼中。當(dāng)執(zhí)行foreach時,可能子進(jìn)程已經(jīng)全部執(zhí)行完畢并退出。pcntl_waitpid會立刻返回,連續(xù)輸出三個parent。

(*在子進(jìn)程中,需通過exit來退出,不然會產(chǎn)生遞歸多進(jìn)程,父進(jìn)程中不需要exit,不然會中斷多進(jìn)程。)

例6如下:

<?phpdefine(’FORK_NUMS’, 3);$pids = array();$fp = fopen(’./test.log’, ’wb’);$num = 1;for($i = 0; $i < FORK_NUMS; ++$i) { $pids[$i] = pcntl_fork(); if($pids[$i] == -1) { die(’fork error’); } else if ($pids[$i]) { } else { for($i = 0; $i < 5; ++$i) { flock($fp, LOCK_EX); fwrite($fp, getmypid() . ’ : ’ . date(’Y-m-d H:i:s’) . ' : {$num} rn'); flock($fp, LOCK_UN); echo getmypid(), ': success rn'; ++$num; } exit; }}foreach($pids as $k => $v) { if($v) { pcntl_waitpid($v, $status); }}fclose($fp);

代碼如上:我們創(chuàng)建三個子進(jìn)程,來同時向test.log文件寫入內(nèi)容,test.log內(nèi)容如下:

19507 : 2016-03-16 20:40:52 : 119507 : 2016-03-16 20:40:52 : 219507 : 2016-03-16 20:40:52 : 319507 : 2016-03-16 20:40:52 : 419507 : 2016-03-16 20:40:52 : 519509 : 2016-03-16 20:40:52 : 119509 : 2016-03-16 20:40:52 : 219509 : 2016-03-16 20:40:52 : 319509 : 2016-03-16 20:40:52 : 419509 : 2016-03-16 20:40:52 : 519508 : 2016-03-16 20:40:52 : 119508 : 2016-03-16 20:40:52 : 219508 : 2016-03-16 20:40:52 : 319508 : 2016-03-16 20:40:52 : 419508 : 2016-03-16 20:40:52 : 5

我們可以看到三個子進(jìn)程的pid,它們分別執(zhí)行了5次,時間幾乎是在同時。但是$num的值并沒像我們期望的那樣從1-15進(jìn)行遞增。子進(jìn)程中的變量是各自獨立的,互不影響。子進(jìn)程會自動復(fù)制父進(jìn)程空間里的變量。

如何在進(jìn)程中共享數(shù)據(jù)?

我們通過php的共享內(nèi)存函數(shù)shmop來實現(xiàn)。

<?phpdefine(’FORK_NUMS’, 3);$pids = array();$fp = fopen(’./test.log’, ’wb’);$num = 1;//共享內(nèi)存段的key$shmKey = 123;//創(chuàng)建共享內(nèi)存段$shmId = shmop_open($shmKey, ’c’, 0777, 64);//寫入數(shù)據(jù)到共享內(nèi)存段shmop_write($shmId, $num, 0);for($i = 0; $i < FORK_NUMS; ++$i) { $pids[$i] = pcntl_fork(); if($pids[$i] == -1) { die(’fork error’); } else if ($pids[$i]) { //阻塞,等待子進(jìn)程退出 //注意這里,如果是非阻塞的話,$num的計數(shù)會出現(xiàn)問題。 pcntl_waitpid($pids[$i], $status); } else { //讀取共享內(nèi)存段中的數(shù)據(jù) $num = shmop_read($shmId, 0, 64); for($i = 0; $i < 5; ++$i) { fwrite($fp, getmypid() . ’ : ’ . date(’Y-m-d H:i:s’) . ' : {$num} rn'); echo getmypid(), ': success rn'; //遞增$num $num = intval($num) + 1; } //寫入到共享內(nèi)存段中 shmop_write($shmId, $num, 0); exit; }}//shmop_delete不會實際刪除該內(nèi)存段,它將該內(nèi)存段標(biāo)記為刪除。shmop_delete($shmId);shmop_close($shmId);fclose($fp);

上述代碼的運行結(jié)果如下:

19923 : 2016-03-17 00:05:18 : 119923 : 2016-03-17 00:05:18 : 219923 : 2016-03-17 00:05:18 : 319923 : 2016-03-17 00:05:18 : 419923 : 2016-03-17 00:05:18 : 519924 : 2016-03-17 00:05:18 : 619924 : 2016-03-17 00:05:18 : 719924 : 2016-03-17 00:05:18 : 819924 : 2016-03-17 00:05:18 : 919924 : 2016-03-17 00:05:18 : 1019925 : 2016-03-17 00:05:18 : 1119925 : 2016-03-17 00:05:18 : 1219925 : 2016-03-17 00:05:18 : 1319925 : 2016-03-17 00:05:18 : 1419925 : 2016-03-17 00:05:18 : 15

這樣我們就在進(jìn)程間共享了$num的數(shù)據(jù)。

更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《PHP進(jìn)程與線程操作技巧總結(jié)》、《PHP網(wǎng)絡(luò)編程技巧總結(jié)》、《PHP基本語法入門教程》、《PHP數(shù)組(Array)操作技巧大全》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫操作入門教程》及《php常見數(shù)據(jù)庫操作技巧匯總》

希望本文所述對大家PHP程序設(shè)計有所幫助。

標(biāo)簽: PHP
相關(guān)文章:
主站蜘蛛池模板: 国产成人在线视频观看 | 中文精品爱久久久国产 | 日韩黄在线观看免费视频 | 精品手机在线视频 | 欧美一区二区视频 | 小明台湾成人永久免费看看 | 日本高清色本在线www游戏 | 精品久久久久久久久久久 | 成人欧美一区二区三区视频xxx | 欧美一级高清免费播放 | 国产一区在线播放 | 一区二三区国产 | 国产17部性孕妇孕交在线 | 福利三区 | 久久久久久91香蕉国产 | 国产 一二三四五六 | 另类在线视频 | 韩国日本三级在线观看 | 欧美在线视频看看 | 亚洲成a人一区二区三区 | 国产精品私人玩物在线观看 | 国产精品亚洲专一区二区三区 | 女人精aaaa片一级毛片女女 | 毛片免费网址 | 日韩成人免费一级毛片 | 色一级片 | 免费一区二区三区四区五区 | 久久91精品牛牛 | 国产欧美日韩一区二区三区 | 亚洲精品综合一区二区 | 亚洲精彩视频在线观看 | 国产成人免费在线观看 | 在线观看精品视频网站www | 亚洲 欧美 日韩在线 | 国语自产拍天天在线 | 国产日产高清欧美一区二区三区 | 中文字幕在线无限2021 | 国产一级做a爱免费观看 | 天天躁天天碰天天看 | 亚洲黄色第一页 | 一级一级一片在线观看 |