濮阳杆衣贸易有限公司

主頁 > 知識(shí)庫 > php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法分析

php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法分析

熱門標(biāo)簽:青島語音外呼系統(tǒng)招商 河南電銷卡外呼系統(tǒng)哪家強(qiáng) 岳陽外呼型呼叫中心系統(tǒng)在哪里 昭通辦理400電話 揚(yáng)州地圖標(biāo)注app 百應(yīng)電話機(jī)器人服務(wù) 山西回?fù)芡夂粝到y(tǒng) 騰訊外呼管理系統(tǒng) 山西探意電話機(jī)器人

本文實(shí)例講述了php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法。分享給大家供大家參考,具體如下:

對(duì)于用PHP進(jìn)行多進(jìn)程并發(fā)編程,不可避免要遇到僵尸進(jìn)程的問題。

僵尸進(jìn)程是指的父進(jìn)程已經(jīng)退出,而該進(jìn)程dead之后沒有進(jìn)程接受,就成為僵尸進(jìn)程(zombie)進(jìn)程。任何進(jìn)程在退出前(使用exit退出) 都會(huì)變成僵尸進(jìn)程(用于保存進(jìn)程的狀態(tài)等信息),然后由init進(jìn)程接管。如果不及時(shí)回收僵尸進(jìn)程,那么它在系統(tǒng)中就會(huì)占用一個(gè)進(jìn)程表項(xiàng),如果這種僵尸進(jìn)程過多,最后系統(tǒng)就沒有可以用的進(jìn)程表項(xiàng),于是也無法再運(yùn)行其它的程序。 

方法一:

父進(jìn)程通過pcntl_wait和pcntl_waitpid等函數(shù)等待子進(jìn)程結(jié)束

$pid = pcntl_fork();

if($pid == -1) {
  die('fork error');
} else if ($pid) {
  //父進(jìn)程阻塞著等待子進(jìn)程的退出
  //pcntl_wait($status);

  //pcntl_waitpid($pid, $status);
  
  //非阻塞方式
  //pcntl_wait($status, WNOHANG);

  //pcntl_waitpid($pid, $status, WNOHANG);
} else {
  sleep(3);
  echo "child \r\n";
  exit;
}

方法二:

可以用signal函數(shù)為SIGCHLD安裝handler,因?yàn)樽舆M(jìn)程結(jié)束后,父進(jìn)程會(huì)收到該信號(hào),可以在handler中調(diào)用pcntl_wait或pcntl_waitpid來回收。

?php
declare(ticks = 1);

//信號(hào)處理函數(shù)
function sig_func() {
  echo "SIGCHLD \r\n";
  pcntl_wait($status);

  //pcntl_waitpid(-1, $status);

  //非阻塞
  //pcntl_wait($status, WNOHANG);
  //pcntl_waitpid(-1, $status, WNOHANG);
}

pcntl_signal(SIGCHLD, 'sig_func');

$pid = pcntl_fork();

if($pid == -1) {
  die('fork error');
} else if ($pid) {
  sleep(10);
} else {
  sleep(3);
  echo "child \r\n";
  exit;
}

如果子進(jìn)程還沒有結(jié)束時(shí),父進(jìn)程就結(jié)束了,那么init進(jìn)程會(huì)自動(dòng)接手這個(gè)子進(jìn)程,進(jìn)行回收。

如果父進(jìn)程是循環(huán),又沒有安裝SIGCHLD信號(hào)處理函數(shù)調(diào)用wait或waitpid()等待子進(jìn)程結(jié)束。那么子進(jìn)程結(jié)束后,沒有回收,就產(chǎn)生僵尸進(jìn)程了。 

例如:

?php
$pid = pcntl_fork();

if($pid == -1) {
  die('fork error');
} else if ($pid) {
  for(;;) {
    sleep(3);
  }
} else {
  echo "child \r\n";
  exit;
}

父進(jìn)程是個(gè)死循環(huán),也沒有安裝SIGCHLD信號(hào)處理函數(shù),子進(jìn)程結(jié)束后。我們通過如下命令查看

> ps -A -o stat,ppid,pid,cmd | grep -e '^[Zz]'

會(huì)發(fā)現(xiàn)一個(gè)僵尸進(jìn)程。 

代碼改進(jìn)一下:

?php
declare(ticks = 1);

//信號(hào)處理函數(shù)
function sig_func() {
  echo "SIGCHLD \r\n";

  pcntl_waitpid(-1, $status, WNOHANG);
}

pcntl_signal(SIGCHLD, 'sig_func');

$pid = pcntl_fork();

if($pid == -1) {
  die('fork error');
} else if ($pid) {
  for(;;) {
    sleep(3);
  }
} else {
  echo "child \r\n";
  exit;
}

當(dāng)子進(jìn)程結(jié)束后,再通過命令查看時(shí),我們發(fā)現(xiàn)這時(shí)就沒有僵尸進(jìn)程了,這說明父進(jìn)程對(duì)它進(jìn)行了回收。 

方法三:

如果父進(jìn)程不關(guān)心子進(jìn)程什么時(shí)候結(jié)束,那么可以用pcntl_signal(SIGCHLD, SIG_IGN)通知內(nèi)核,自己對(duì)子進(jìn)程的結(jié)束不感興趣,那么子進(jìn)程結(jié)束后,內(nèi)核會(huì)回收,并不再給父進(jìn)程發(fā)送信號(hào)。

?php
declare(ticks = 1);

pcntl_signal(SIGCHLD, SIG_IGN);

$pid = pcntl_fork();

if($pid == -1) {
  die('fork error');
} else if ($pid) {
  for(;;) {
    sleep(3);
  }
} else {
  echo "child \r\n";
  exit;
}

當(dāng)子進(jìn)程結(jié)束后,SIGCHLD信號(hào)并不會(huì)發(fā)送給父進(jìn)程,而是通知內(nèi)核對(duì)子進(jìn)程進(jìn)行了回收。 

方法四:

通過pcntl_fork兩次,也就是父進(jìn)程fork出子進(jìn)程,然后子進(jìn)程中再fork出孫進(jìn)程,這時(shí)子進(jìn)程退出。那么init進(jìn)程會(huì)接管孫進(jìn)程,孫進(jìn)程退出后,init會(huì)回收。不過子進(jìn)程還是需要父進(jìn)程進(jìn)行回收。我們把業(yè)務(wù)邏輯放到孫進(jìn)程中執(zhí)行,父進(jìn)程就不需要pcntl_wait或pcntl_waitpid來等待孫進(jìn)程(即業(yè)務(wù)進(jìn)程)。

?php
$pid = pcntl_fork();

if($pid == -1) {
  die('fork error');
} else if ($pid) {
  //父進(jìn)程等待子進(jìn)程退出
  pcntl_wait($status);
  echo "parent \r\n";
} else {
  //子進(jìn)程再fork一次,產(chǎn)生孫進(jìn)程
  $cpid = pcntl_fork();  
  if($cpid == -1) {
    die('fork error');
  } else if ($cpid) {
    //這里是子進(jìn)程,直接退出
    echo "child \r\n";
    exit;
  } else {
    //這里是孫進(jìn)程,處理業(yè)務(wù)邏輯
    for($i = 0; $i  10; ++$i) {
      echo "work... \r\n";
      sleep(3);
    }
  }
}

子進(jìn)程退出后,父進(jìn)程回收子進(jìn)程,孫進(jìn)程繼續(xù)業(yè)務(wù)邏輯的處理。當(dāng)孫進(jìn)程也執(zhí)行完畢退出后,init回收孫進(jìn)程。

更多關(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ù)庫操作技巧匯總》

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

您可能感興趣的文章:
  • php多進(jìn)程中的阻塞與非阻塞操作實(shí)例分析
  • php 的多進(jìn)程操作實(shí)踐案例分析
  • php 多進(jìn)程編程父進(jìn)程的阻塞與非阻塞實(shí)例分析
  • php實(shí)現(xiàn)的簡單多進(jìn)程服務(wù)器類完整示例
  • PHP多進(jìn)程簡單實(shí)例小結(jié)
  • PHP 多進(jìn)程與信號(hào)中斷實(shí)現(xiàn)多任務(wù)常駐內(nèi)存管理實(shí)例方法
  • php swoole多進(jìn)程/多線程用法示例【基于php7nts版】
  • PHP基于swoole多進(jìn)程操作示例
  • 詳解PHP多進(jìn)程消費(fèi)隊(duì)列

標(biāo)簽:湛江 寶雞 銅川 黃南 鎮(zhèn)江 宜賓 南陽 婁底

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法分析》,本文關(guān)鍵詞  php,多,進(jìn)程,并發(fā),編程,防止,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法分析》相關(guān)的同類信息!
  • 本頁收集關(guān)于php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法分析的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    察隅县| 鲁甸县| 德惠市| 银川市| 措勤县| 高淳县| 涿鹿县| 乐亭县| 岳阳市| 开远市| 黄浦区| 定西市| 于都县| 桃园市| 寻乌县| 隆安县| 丹凤县| 巴东县| 仲巴县| 宣武区| 昆山市| 通榆县| 汝州市| 嘉义市| 海林市| 临湘市| 临邑县| 贺州市| 遂昌县| 大庆市| 苗栗市| 宾川县| 贵德县| 安陆市| 托克托县| 大理市| 英吉沙县| 仁寿县| 祥云县| 乌兰浩特市| 宝兴县|