欧美日韩精品在线,国内精品久久久久久久久,一级毛片恃级毛片直播,清纯唯美亚洲综合欧美色

深入解析PHP中的(偽)多線程與多進(jìn)程_PHP教程

編輯Tag賺U幣
教程Tag:暫無Tag,歡迎添加,賺取U幣!

推薦:php 常用算法和時間復(fù)雜度
本篇文章是對php中的常用算法以及時間復(fù)雜度進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下 按數(shù)量級遞增排列,常見的時間復(fù)雜度有:常數(shù)階O(1),對數(shù)階O(log2n),線性階O(n),線性對數(shù)階O(nlog2n),平方階O(n2),立方階O(n3) 復(fù)制代碼 代碼如下: //二分查找O(log2n) function erf

本篇文章是對PHP中的(偽)多線程與多進(jìn)程進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下  

(偽)多線程:借助外力
利用WEB服務(wù)器本身的多線程來處理,從WEB服務(wù)器多次調(diào)用我們需要實(shí)現(xiàn)多線程的程序。
QUOTE:
我們知道PHP本身是不支持多線程的, 但是我們的WEB服務(wù)器是支持多線程的.
也就是說可以同時讓多人一起訪問. 這也是我在PHP中實(shí)現(xiàn)多線程的基礎(chǔ).
假設(shè)我們現(xiàn)在運(yùn)行的是a.php這個文件. 但是我在程序中又請求WEB服務(wù)器運(yùn)行另一個b.php
那么這兩個文件將是同時執(zhí)行的.
(PS: 一個鏈接請求發(fā)送之后, WEB服務(wù)器就會執(zhí)行它, 而不管客戶端是否已經(jīng)退出)
有些時候, 我們想運(yùn)行的不是另一個文件, 而是本文件中的一部分代碼.該怎么辦呢?
其實(shí)可是通過參數(shù)來控制a.php來運(yùn)行哪一段程序.
下面看一個例子:

復(fù)制代碼 代碼如下:
<?php
function runThread(){
$fp = fsockopen('localhost', 80, $errno, $errmsg);
fputs($fp, "GET /a.php?act=brnrn");//這里的第二個參數(shù)是HTTP協(xié)議中規(guī)定的請求頭,不明白的請看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 比較一下兩個文件的中訪問的時間. 大家會發(fā)現(xiàn), 這兩個的確是在不同線程中運(yùn)行的.有些時間完全一樣.
上面只是一個簡單的例子, 大家可以改進(jìn)成其它形式.
既然PHP中也能多線程了, 那么問題也來了, 那就是同步的問題. 我們知道 PHP本身是不支持多線程的. 所以更不會有什么像Java 中synchronize的方法了. 那我們該如何做呢.

1. 盡量不訪問同一個資源. 以避免沖突. 但是可以同時像數(shù)據(jù)庫操作. 因?yàn)閿?shù)據(jù)庫是支持并發(fā)操作的. 所以在多線程的PHP中
不 要向同一個文件中寫入數(shù)據(jù). 如果必須要寫的話, 用別的方法進(jìn)行同步.. 如調(diào)用 flock對文件進(jìn)行加鎖等. 或建立臨時文件并在另外的線程中等待這個文件的消失 while(file_exits('xxx')); 這樣就等于這個臨時文件存在時, 表示其實(shí)線程正在操作,如果沒有了這個文件, 說明其它線程已經(jīng)釋放了這個.

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

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

本人做的一個能從8848、soaso這些商城網(wǎng)站搜索信息的程序。還有一個從阿里巴巴網(wǎng)站上讀取商業(yè)信息和公司目錄的程序也用到了此技術(shù)。 因?yàn)檫@兩個程序都是要不斷的鏈接它們的服務(wù)器讀取信息并保存到數(shù)據(jù)庫。 利用此技術(shù)正好消除了在等待響應(yīng)時的瓶頸。

多進(jìn)程:使用PHP的Process Control Functions(PCNTL/線程控制函數(shù))
只能用在Unix Like OS,Windows不可用。
編譯php的時候,需要加上--enable-pcntl,且推薦僅僅在CLI模式運(yùn)行,不要在WEB服務(wù)器環(huán)境運(yùn)行。
以下為簡短的測試代碼:

復(fù)制代碼 代碼如下:
declare(ticks=1);
$bWaitFlag = FALSE; /// 是否等待進(jìn)程結(jié)束
$intNum = 10; /// 進(jìn)程總數(shù)
$pids = array(); /// 進(jìn)程PID數(shù)組
echo ("Start\n");
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 ("End\n");


運(yùn)行結(jié)果如下:
CODE:[Copy toclipboard][qiao@oicq qiao]$ phptest.php
Start
End
[qiao@oicq qiao]$ ps -aux | grep "php"
qiao 32275 0.0 0.5 49668 6148pts/1 S 14:03 0:00/usr/local/php4/b
qiao 32276 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
qiao 32277 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
qiao 32278 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
qiao 32279 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
qiao 32280 0.0 0.5 49668 6152pts/1 S 14:03 0:00 /usr/local/php4/b
qiao 32281 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
qiao 32282 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
qiao 32283 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
qiao 32284 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
qiao 32286 0.0 0.0 1620 600pts/1 S 14:03 0:00 grep php
[qiao@oicq qiao]$ 0 -> 1133503401
1 -> 1133503402 *
2 -> 1133503403 **
3 -> 1133503404 ***
4 -> 1133503405 ****
5 -> 1133503406 *****
6 -> 1133503407 ******
7 -> 1133503408 *******
8 -> 1133503409 ********
9 -> 1133503410 *********
[qiao@oicq qiao]$
如果$bWaitFlag=TURE,則結(jié)果如下:
CODE:[Copy toclipboard][qiao@oicq qiao]$ phptest.php
Start
0 -> 1133503602
wait 0 -> 1133503602
1 -> 1133503603 *
wait 1 -> 1133503603
2 -> 1133503604 **
wait 2 -> 1133503604
3 -> 1133503605 ***
wait 3 -> 1133503605
4 -> 1133503606 ****
wait 4 -> 1133503606
5 -> 1133503607 *****
wait 5 -> 1133503607
6 -> 1133503608 ******
wait 6 -> 1133503608
7 -> 1133503609 *******
wait 7 -> 1133503609
8 -> 1133503610 ********
wait 8 -> 1133503610
9 -> 1133503611 *********
wait 9 -> 1133503611
End
[qiao@oicq qiao]$
從 多進(jìn)程的例子可以看出,使用pcntl_fork()之后,將生成一個子進(jìn)程,而且子進(jìn)程運(yùn)行的代碼,從pcntl_fork()之后的代碼開始,而子進(jìn) 程不繼承父進(jìn)程的數(shù)據(jù)信息(實(shí)際上是把父進(jìn)程的數(shù)據(jù)做了一個全新的拷貝),因而使用if(!$pids[$i]) 來控制子進(jìn)程實(shí)際運(yùn)行的代碼段。
更詳細(xì)的研究出于時間關(guān)系,暫時沒有進(jìn)行,你可以參考我給出的手冊的鏈接。

分享:解析PHP跳出循環(huán)的方法以及continue、break、exit的區(qū)別介紹
PHP中的循環(huán)結(jié)構(gòu)大致有for循環(huán),while循環(huán),do{} while 循環(huán)以及foreach循環(huán)幾種,不管哪種循環(huán)中,在 PHP中跳出循環(huán)大致有這么幾種方式: 代碼: 復(fù)制代碼 代碼如下: ?php $i = 1; while (true) { // 這里看上去這個循環(huán)會一直執(zhí)行 if ($i==2) {// 2跳過不顯示 $i++;

共3頁上一頁123下一頁
來源:模板無憂//所屬分類:PHP教程/更新時間:2013-07-02
相關(guān)PHP教程