婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > php多進程并發編程防止出現僵尸進程的方法分析

php多進程并發編程防止出現僵尸進程的方法分析

熱門標簽:青島語音外呼系統招商 河南電銷卡外呼系統哪家強 岳陽外呼型呼叫中心系統在哪里 昭通辦理400電話 揚州地圖標注app 百應電話機器人服務 山西回撥外呼系統 騰訊外呼管理系統 山西探意電話機器人

本文實例講述了php多進程并發編程防止出現僵尸進程的方法。分享給大家供大家參考,具體如下:

對于用PHP進行多進程并發編程,不可避免要遇到僵尸進程的問題。

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

方法一:

父進程通過pcntl_wait和pcntl_waitpid等函數等待子進程結束

$pid = pcntl_fork();

if($pid == -1) {
  die('fork error');
} else if ($pid) {
  //父進程阻塞著等待子進程的退出
  //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函數為SIGCHLD安裝handler,因為子進程結束后,父進程會收到該信號,可以在handler中調用pcntl_wait或pcntl_waitpid來回收。

?php
declare(ticks = 1);

//信號處理函數
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;
}

如果子進程還沒有結束時,父進程就結束了,那么init進程會自動接手這個子進程,進行回收。

如果父進程是循環,又沒有安裝SIGCHLD信號處理函數調用wait或waitpid()等待子進程結束。那么子進程結束后,沒有回收,就產生僵尸進程了。 

例如:

?php
$pid = pcntl_fork();

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

父進程是個死循環,也沒有安裝SIGCHLD信號處理函數,子進程結束后。我們通過如下命令查看

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

會發現一個僵尸進程。 

代碼改進一下:

?php
declare(ticks = 1);

//信號處理函數
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;
}

當子進程結束后,再通過命令查看時,我們發現這時就沒有僵尸進程了,這說明父進程對它進行了回收。 

方法三:

如果父進程不關心子進程什么時候結束,那么可以用pcntl_signal(SIGCHLD, SIG_IGN)通知內核,自己對子進程的結束不感興趣,那么子進程結束后,內核會回收,并不再給父進程發送信號。

?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;
}

當子進程結束后,SIGCHLD信號并不會發送給父進程,而是通知內核對子進程進行了回收。 

方法四:

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

?php
$pid = pcntl_fork();

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

子進程退出后,父進程回收子進程,孫進程繼續業務邏輯的處理。當孫進程也執行完畢退出后,init回收孫進程。

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

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

您可能感興趣的文章:
  • php多進程中的阻塞與非阻塞操作實例分析
  • php 的多進程操作實踐案例分析
  • php 多進程編程父進程的阻塞與非阻塞實例分析
  • php實現的簡單多進程服務器類完整示例
  • PHP多進程簡單實例小結
  • PHP 多進程與信號中斷實現多任務常駐內存管理實例方法
  • php swoole多進程/多線程用法示例【基于php7nts版】
  • PHP基于swoole多進程操作示例
  • 詳解PHP多進程消費隊列

標簽:湛江 寶雞 銅川 黃南 鎮江 宜賓 南陽 婁底

巨人網絡通訊聲明:本文標題《php多進程并發編程防止出現僵尸進程的方法分析》,本文關鍵詞  php,多,進程,并發,編程,防止,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《php多進程并發編程防止出現僵尸進程的方法分析》相關的同類信息!
  • 本頁收集關于php多進程并發編程防止出現僵尸進程的方法分析的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 镇坪县| 奉贤区| 吴堡县| 桦甸市| 时尚| 克山县| 株洲县| 安化县| 苍溪县| 高要市| 得荣县| 永定县| 化州市| 邵武市| 大姚县| 缙云县| 固原市| 离岛区| 信宜市| 温宿县| 大城县| 南平市| 永定县| 格尔木市| 彰武县| 眉山市| 泌阳县| 栾川县| 通榆县| 怀安县| 阿克陶县| 沙雅县| 永济市| 临漳县| 定南县| 洞头县| 青海省| 宁远县| 林州市| 恭城| 惠来县|