萬盛學電腦網

 萬盛學電腦網 >> 網絡編程 >> php編程 >> PHP實現多線程的兩個方法

PHP實現多線程的兩個方法

 PHP+shell實現多線程的方法

先寫個簡單的php代碼,這裡為了讓腳本執行時間更長,方便看效果,sleep一下,呵呵!先看下test.php的代碼:ls

PHP代碼:

<?php
for ($i=0;$i<10;$i++) {
  echo $i;
  sleep(10);
}
?>

在看下shell腳本的代碼,非常簡單

#!/bin/bash
for i in 1 2 3 4 5 6 7 8 9 10
do
  /usr/bin/php -q /var/www/html/test.php &
done

注意到在請求php代碼的那行有一個&符號嗎,這個是關鍵,不加的話是不能進行多線程的,&表示講服務推送到後台執行,因此,在 shell的每次的循環中不必等php的代碼全部執行完在請求下一個文件,而是同時進行的,這樣就實現了多線程,下面運行下shell看下效果,這裡你將 看到10個test.php進程再跑,再利用linux的定時器,定時請求這個shell,在處理一些需要多線程的任務,例如,批量下載時,非常好用!

 

php中用WEB服務器實現多線程

假設我們現在運行的是a.php這個文件. 但是我在程序中又請求WEB服務器運行另一個b.php,那麼這兩個文件將是同時執行的.(PS: 一個鏈接請求發送之後, WEB服務器就會執行它, 而不管客戶端是否已經退出)

有些時候, 我們想運行的不是另一個文件, 而是本文件中的一部分代碼.該怎麼辦呢?
其實可是通過參數來控制a.php來運行哪一段程序.

下面看一個例子:

//a.php,b.php

PHP代碼:--------------------------------------------------------------------------------

<?php
    function runThread()
    {
        $fp = fsockopen('localhost', 80, $errno, $errmsg);
        
        fputs($fp, "GET /b.php?act=brnrn");        //這裡的第二個參數是HTTP協議中規定的請求頭
                                //不明白的請看RFC中的定義
        
        fclose($fp);
    }
 
    function a()
    {
        $fp = fopen('result_a.log', 'w');
        fputs($fp, 'Set in ' . Date('h:i:s', time()) . (double)microtime() . "rn");
        
        fclose($fp);        
    }
 
    function b()
    {
        $fp = fopen('result_b.log', 'w');
        fputs($fp, 'Set in ' . Date('h:i:s', time()) . (double)microtime() . "rn");
        
        fclose($fp);        
    }
 
    if(!isset($_GET['act'])) $_GET['act'] = 'a';
    
    if($_GET['act'] == 'a')
    {
        runThread();
        a();
    }
    else if($_GET['act'] == 'b') b();
?>
--------------------------------------------------------------------------------


打開result_a.log 和 result_b.log 比較一下兩個文件的中訪問的時間. 大家會發現, 這兩個的確是在不同線程中運行的.有些時間完全一樣.

上面只是一個簡單的例子, 大家可以改進成其它形式.


既然PHP中也能多線程了, 那麼問題也來了, 那就是同步的問題. 我們知道 PHP本身是不支持多線程的. 所以更不會有什麼像Java 中synchronize的方法了. 那我們該如何做呢.

1. 盡量不訪問同一個資源. 以避免沖突. 但是可以同時像數據庫操作. 因為數據庫是支持並發操作的. 所以在多線程的PHP中不要向同一個文件中寫入數據. 如果必須要寫的話, 用別的方法進行同步.. 如調用 flock對文件進行加鎖等. 或建立臨時文件並在另外的線程中等待這個文件的消失 while(file_exits('xxx')); 這樣就等於這個臨時文件存在時, 表示其實線程正在操作

如果沒有了這個文件, 說明其它線程已經釋放了這個.

2. 盡量不要從runThread在執行fputs後取這個socket中讀取數據. 因為要實現多線程, 需要的用非阻塞模式. 即在像fgets這樣的函數時立即返回.. 所以讀寫數據就會出問題. 如果使用阻塞模式的話, 程序就不算是多線程了. 他要等上面的返回才執行下面的程序. 所以如果需要交換數據最後利用外面文件或數據中完成. 實在想要的話就用socket_set_nonblock($fp) 來實現.


說了這麼多, 倒底這個有沒有實際的意義呢? 在什麼時候需要這種用這種方法呢 ?
答案是肯定的. 大家知道. 在一個不斷讀取網絡資源的應用中, 網絡的速度是瓶頸. 如果采多這種形式就可以同時以多個線程對不同的頁面進行讀取.

本人做的一個能從8848、soaso這些商城網站搜索信息的程序。還有一個從阿裡巴巴網站上讀取商業信息和公司目錄的程序也用到了此技術。 因為這兩個程序都是要不斷的鏈接它們的服務器讀取信息並保存到數據庫。 利用此技術正好消除了在等待響應時的瓶頸。

 


php模擬實現多線程的三種方法

PHP語言本身是不支持多線程的. 總結了一下網上關於PHP模擬多線程的方法, 總的來說, 都是利用了PHP的好伙伴們本身所具有的多線程能力. PHP的好伙伴指的就是LINUX和APACHE啦, LAMP嘛.

  另外, 既然是模擬的, 就不是真正的多線程. 其實只是多進程. 進程和線程是兩個不同的概念. 好了, 以下方法都是從網上找來的.

  1. 利用LINUX操作系統

<?php
for ($i=0;$i<10;$i++) {
  echo $i;
  sleep(5);
}
?>

上面存成test.php, 然後寫一段SHELL代碼

#!/bin/bash
for i in 1 2 3 4 5 6 7 8 9 10
do
  php -q test.php &
done

2. 利用fork子進程(其實同樣是利用LINUX操作系統)

<?php
declare(ticks=1);
$bWaitFlag = FALSE; /// 是否等待進程結束
$intNum = 10;      /// 進程總數
$pids = array();    /// 進程PID數組
echo ("Startn");
for($i = 0; $i < $intNum; $i++) {
 $pids[$i] = pcntl_fork();/// 產生子進程,而且從當前行之下開試運行代碼,而且不繼承父進程的數據信息
 if(!$pids[$i]) {
  // 子進程進程代碼段_Start
  $str="";
  sleep(5+$i);
  for ($j=0;$j<$i;$j++) {$str.="*";}
  echo "$i -> " . time() . " $str n";
  exit();
  // 子進程進程代碼段_End
 }
}
if ($bWaitFlag)
{
 for($i = 0; $i < $intNum; $i++) {
  pcntl_waitpid($pids[$i], $status, WUNTRACED);
  echo "wait $i -> " . time() . "n";
 }
}
echo ("Endn");
?>

3. 利用WEB SERVER, PHP不支持多線程, APACHE可是支持的, 呵呵.

假設我們現在運行的是a.php這個文檔. 但是我在程式中又請求WEB服務器運行另一個b.php

那麼這兩個文檔將是同時執行的.(代碼同上)

當然啦,也可以把需要多線程處理的部分交給JAVA去處理, 然後在PHP裡調用, 哈哈.

<?php
system('java multiThread.java');
?>

copyright © 萬盛學電腦網 all rights reserved