PHP+shell實現(xiàn)多線程的方法
先寫個簡單的php代碼,這里為了讓腳本執(zhí)行時間更長,方便看效果,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代碼的那行有一個&符號嗎,這個是關鍵,不加的話是不能進行多線程的,&表示講服務推送到后臺執(zhí)行,因此,在shell的每次的循環(huán)中不必等php的代碼全部執(zhí)行完在請求下一個文件,而是同時進行的,這樣就實現(xiàn)了多線程,下面運行下shell看下效果,這里你將看到10個test.php進程再跑,再利用linux的定時器,定時請求這個shell,在處理一些需要多線程的任務,例如,批量下載時,非常好用!
php中用WEB服務器實現(xiàn)多線程
假設我們現(xiàn)在運行的是a.php這個文件.但是我在程序中又請求WEB服務器運行另一個b.php,那么這兩個文件將是同時執(zhí)行的.(PS:一個鏈接請求發(fā)送之后,WEB服務器就會執(zhí)行它,而不管客戶端是否已經(jīng)退出)
有些時候,我們想運行的不是另一個文件,而是本文件中的一部分代碼.該怎么辦呢?
其實可是通過參數(shù)來控制a.php來運行哪一段程序.
下面看一個例子:
//a.php,b.php
PHP代碼:--------------------------------------------------------------------------------
functionrunThread()
{
$fp=fsockopen('localhost',80,$errno,$errmsg);
fputs($fp,"GET/b.php?act=b\r\n\r\n");//這里的第二個參數(shù)是HTTP協(xié)議中規(guī)定的請求頭
//不明白的請看RFC中的定義
fclose($fp);
}
functiona()
{
$fp=fopen('result_a.log','w');
fputs($fp,'Setin'.Date('h:i:s',time()).(double)microtime()."\r\n");
fclose($fp);
}
functionb()
{
$fp=fopen('result_b.log','w');
fputs($fp,'Setin'.Date('h:i:s',time()).(double)microtime()."\r\n");
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比較一下兩個文件的中訪問的時間.大家會發(fā)現(xiàn),這兩個的確是在不同線程中運行的.有些時間完全一樣.
上面只是一個簡單的例子,大家可以改進成其它形式.
既然PHP中也能多線程了,那么問題也來了,那就是同步的問題.我們知道PHP本身是不支持多線程的.所以更不會有什么像Java中synchronize的方法了.那我們該如何做呢.
1.盡量不訪問同一個資源.以避免沖突.但是可以同時像數(shù)據(jù)庫操作.因為數(shù)據(jù)庫是支持并發(fā)操作的.所以在多線程的PHP中不要向同一個文件中寫入數(shù)據(jù).如果必須要寫的話,用別的方法進行同步..如調(diào)用flock對文件進行加鎖等.或建立臨時文件并在另外的線程中等待這個文件的消失while(file_exits('xxx'));這樣就等于這個臨時文件存在時,表示其實線程正在操作
如果沒有了這個文件,說明其它線程已經(jīng)釋放了這個.
2.盡量不要從runThread在執(zhí)行fputs后取這個socket中讀取數(shù)據(jù).因為要實現(xiàn)多線程,需要的用非阻塞模式.即在像fgets這樣的函數(shù)時立即返回..所以讀寫數(shù)據(jù)就會出問題.如果使用阻塞模式的話,程序就不算是多線程了.他要等上面的返回才執(zhí)行下面的程序.所以如果需要交換數(shù)據(jù)最后利用外面文件或數(shù)據(jù)中完成.實在想要的話就用socket_set_nonblock($fp)來實現(xiàn).
說了這么多,倒底這個有沒有實際的意義呢?在什么時候需要這種用這種方法呢?
答案是肯定的.大家知道.在一個不斷讀取網(wǎng)絡資源的應用中,網(wǎng)絡的速度是瓶頸.如果采多這種形式就可以同時以多個線程對不同的頁面進行讀取.
本人做的一個能從8848、soaso這些商城網(wǎng)站搜索信息的程序。還有一個從阿里巴巴網(wǎng)站上讀取商業(yè)信息和公司目錄的程序也用到了此技術。因為這兩個程序都是要不斷的鏈接它們的服務器讀取信息并保存到數(shù)據(jù)庫。利用此技術正好消除了在等待響應時的瓶頸。
php模擬實現(xiàn)多線程的三種方法
PHP語言本身是不支持多線程的.總結了一下網(wǎng)上關于PHP模擬多線程的方法,總的來說,都是利用了PHP的好伙伴們本身所具有的多線程能力.PHP的好伙伴指的就是LINUX和APACHE啦,LAMP嘛.
另外,既然是模擬的,就不是真正的多線程.其實只是多進程.進程和線程是兩個不同的概念.好了,以下方法都是從網(wǎng)上找來的.
1.利用LINUX操作系統(tǒng)
for($i=0;$i<10;$i++){
echo$i;
sleep(5);
}
?>
上面存成test.php,然后寫一段SHELL代碼
#!/bin/bash
foriin12345678910
do
php-qtest.php&
done
2.利用fork子進程(其實同樣是利用LINUX操作系統(tǒng))
declare(ticks=1);
$bWaitFlag=FALSE;///是否等待進程結束
$intNum=10;///進程總數(shù)
$pids=array();///進程PID數(shù)組
echo("Startn");
for($i=0;$i<$intNum;$i++){
$pids[$i]=pcntl_fork();///產(chǎn)生子進程,而且從當前行之下開試運行代碼,而且不繼承父進程的數(shù)據(jù)信息
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可是支持的,呵呵.
假設我們現(xiàn)在運行的是a.php這個文檔.但是我在程式中又請求WEB服務器運行另一個b.php
那么這兩個文檔將是同時執(zhí)行的.(代碼同上)
當然啦,也可以把需要多線程處理的部分交給JAVA去處理,然后在PHP里調(diào)用,哈哈.
system('javamultiThread.java');
?>
擴展資料:PHP即“超文本預處理器”,是一種通用開源腳本語言。PHP是在服務器端執(zhí)行的腳本語言,與C語言類似,是常用的網(wǎng)站編程語言。PHP獨特的語法混合了C、Java、Perl以及PHP自創(chuàng)的語法。利于學習,使用廣泛,主要適用于Web開發(fā)領域。