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