PHP多進程編程實例

場景:平常任務中,有時須要經過php腳本執行一些日誌分析,隊列處理等任務,當數據量比較大時,可使用多進程來處理。php

準備:php多進程須要pcntl,posix擴展支持,能夠經過 php - m 查看,沒安裝的話須要從新編譯php,加上參數--enable-pcntl,posix通常默認會有。shell

建立子進程的函數fork微信

pcntl_fork — 在當前進程當前位置產生分支(子進程)。譯註:fork是建立了一個子進程,父進程和子進程 都從fork的位置開始向下繼續執行,不一樣的是父進程執行過程當中,獲得的fork返回值爲子進程號,而子進程獲得的是0。函數

一個fork子進程的基礎示例: 學習

 1 <?php
 2 $pid = pcntl_fork();//父進程和子進程都會執行下面代碼
 3 if ($pid == -1) {   
 4      //錯誤處理:建立子進程失敗時返回-1.
 5      die('could not fork');
 6 } else if ($pid) {
 7    //父進程會獲得子進程號,因此這裏是父進程執行的邏輯
 8      pcntl_wait($status); 
 9     //等待子進程中斷,防止子進程成爲殭屍進程。
10 } else {     
11  //子進程獲得的$pid爲0, 因此這裏是子進程執行的邏輯。
12 }

若是一個任務被分解成多個進程執行,就會減小總體的耗時。spa

好比有一個比較大的數據文件要處理,這個文件由不少行組成。若是單進程執行要處理的任務,量很大時要耗時比較久。這時能夠考慮多進程。日誌

多進程處理分解任務,每一個進程處理文件的一部分,這樣須要均分割一下這個大文件成多個小文件(進程數和小文件的個數等同就能夠)。code

好比該文件file.log有10萬行數據,如今想分4個進程處理。須要分割2.5萬行一個文件。命令split能夠作到。blog

split的用法比較簡單,能夠man split查看下手冊。 隊列

split -l 25000 -d file.log prefix_name

-l是按照行分割,-d是分割後的文件名按照數字,-a是分割後的文件個數位數(默認是2,作多就是99個;好比超過100個,-a能夠寫3)。本身嘗試分割一下就知道了。

處理代碼:

<?php
shell_exec('split -l 25000 -d file.log prefix_name');
// 3個子進程處理任務
for ($i = 0; $i < 3; $i++){   
 $pid = pcntl_fork();    
 if ($pid == -1) {
    die("could not fork");
} elseif ($pid) { 
     echo "I'm the Parent $i\n";
 } else {// 子進程處理
  $content = file_get_contents("prefix_name0".$i); 
// 業務處理 begin
// 業務處理 end
    exit;
// 必定要注意退出子進程,不然pcntl_fork() 會被子進程再fork,帶來處理上的影響。    }
}// 等待子進程執行結束
while (pcntl_waitpid(0, $status) != -1) {    
$status = pcntl_wexitstatus($status);    
echo "Child $status completed\n";}

能夠關注微信公衆號 lovephp 一塊兒學習

相關文章
相關標籤/搜索