PHP+shell實現多線程的方法
先寫個簡單的php代碼,這裏為了讓腳本執行時間更長,方便看效果,sleep壹下,呵呵!先看下test.php的代碼:ls
PHP代碼:
for($i=0;$i<10;$i++){
echo$i;
sleep(10);
}
>在看下shell腳本的代碼,非常簡單
#!/bin/bash
foriin12345678910
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代碼:--------------------------------------------------------------------------------
functionrunThread()
{
$fp=fsockopen('localhost',80,$errno,$errmsg);
fputs($fp,\"GET/b.php?act=b\");//這裏的第二個參數是p>
fclose($fp);
}
functiona()
{
$fp=fopen('result_a.log','w');
fputs($fp,'Setin'.Date('h:i:s',time()).(double)microtime().\"\");
fclose($fp);
}
functionb()
{
$fp=fopen('result_b.log','w');
fputs($fp,'Setin'.Date('h:i:s',time()).(double)microtime().\"\");
fclose($fp);
}
if(!isset($_GET['act']))$_GET['act']='a';
if($_GET['act']=='a')
{
runThread();
a();
}
elseif($_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操作系統
for($i=0;$i<10;$i++){
echo$i;
sleep(5);
}
>上面存成test.php,然後寫壹段SHELL代碼
#!/bin/bash
foriin12345678910
do
php-qtest.php&
done
2.利用fork子進程(其實同樣是利用LINUX操作系統)
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().\"$strn\";
exit();
//子進程進程代碼段_End
}
}
if($bWaitFlag)
{
for($i=0;$i<$intNum;$i++){
pcntl_waitpid($pids[$i],$status,WUNTRACED);
echo\"wait$i->\".time().\"n\";
}
}
echo(\"Endn\");
>3.利用WEBSERVER,PHP不支持多線程,APACHE可是支持的,呵呵.
假設我們現在運行的是a.php這個文檔.但是我在程式中又請求WEB服務器運行另壹個b.php
那麽這兩個文檔將是同時執行的.(代碼同上)
當然啦,也可以把需要多線程處理的部分交給JAVA去處理,然後在PHP裏調用,哈哈.
system('javamultiThread.java');
>擴展資料: