《PHP實(shí)戰(zhàn):關(guān)于PHP中協(xié)程和阻塞的一些理解與思考》要點(diǎn):
本文介紹了PHP實(shí)戰(zhàn):關(guān)于PHP中協(xié)程和阻塞的一些理解與思考,希望對(duì)您有用。如果有疑問(wèn),可以聯(lián)系我們。
PHP應(yīng)用前言
PHP應(yīng)用本文主要給大家介紹了關(guān)于PHP中協(xié)程和阻塞的理解與思考,分享出來(lái)供大家參考學(xué)習(xí),下面話不多說(shuō)了,來(lái)一起看看詳細(xì)的介紹:
PHP應(yīng)用進(jìn)程、線程、協(xié)程
PHP應(yīng)用關(guān)于進(jìn)程、線程、協(xié)程,有非常詳細(xì)和豐富的博客或者學(xué)習(xí)資源,我不在此做贅述,我大致在此介紹一下這幾個(gè)東西.
PHP應(yīng)用PHP中的協(xié)程實(shí)現(xiàn)基礎(chǔ) yield
PHP應(yīng)用yield的根本實(shí)現(xiàn)是生成器類,而迭代器類是迭代器接口的實(shí)現(xiàn):
PHP應(yīng)用
Generator implements Iterator {
public mixed current ( void ) // 返回當(dāng)前產(chǎn)生的值
public mixed key ( void ) // 返回當(dāng)前產(chǎn)生的鍵
public void next ( void ) // 生成器繼續(xù)執(zhí)行
public void rewind ( void ) // 重置迭代器,如果迭代已經(jīng)開(kāi)始了,這里會(huì)拋出一個(gè)異常.
// renwind的執(zhí)行將會(huì)導(dǎo)致第一個(gè)yield被執(zhí)行, 并且忽略了他的返回值.
public mixed send ( mixed $value ) // 向生成器中傳入一個(gè)值,并且當(dāng)做 yield 表達(dá)式的結(jié)果,然后繼續(xù)執(zhí)行生成器.如果當(dāng)這個(gè)方法被調(diào)用時(shí),生成器
// 不在 yield 表達(dá)式,那么在傳入值之前,它會(huì)先運(yùn)行到第一個(gè) yield 表達(dá)式.
public void throw ( Exception $exception ) // 向生成器中拋入一個(gè)異常
public bool valid ( void ) // 檢查迭代器是否被關(guān)閉
public void __wakeup ( void ) // 序列化回調(diào),拋出一個(gè)異常以表示生成器不能被序列化.
}
PHP應(yīng)用以上解析可以參考PHP官方文檔.
PHP應(yīng)用http://php.net/manual/zh/clas...
PHP應(yīng)用以及這篇詳細(xì)文檔:
PHP應(yīng)用/article/39424_all.htm
PHP應(yīng)用我就以他實(shí)現(xiàn)的協(xié)程多任務(wù)調(diào)度為基礎(chǔ)做一下例子說(shuō)明并說(shuō)一下關(guān)于我在阻塞方面所做的一些思考.
PHP應(yīng)用自定義簡(jiǎn)單定時(shí)執(zhí)行任務(wù)示例:
PHP應(yīng)用(此例子必須依賴于以上鳥哥實(shí)現(xiàn)的協(xié)程調(diào)度代碼)
PHP應(yīng)用
class timer {
private $start = 0; // 定時(shí)開(kāi)始時(shí)間
private $timer; // 間隔的時(shí)間差,單位秒
private $value = 0; // 產(chǎn)生的結(jié)果值
private $callback; // 異步回調(diào)
private $isEnd = false; // 當(dāng)前定時(shí)器任務(wù)是否結(jié)束
public function __construct($timer,callable $callback)
{
$this->start = time();
$this->timer = $timer;
$this->callback = $callback;
}
public function run() {
if($this->valid()) {
$callback = $this->callback;
$callback($this->value ++,$this);
$this->start = time();
}
}
/**
* 定時(shí)執(zhí)行檢查
*/
public function valid() {
$end = time();
if($end - $this->start >= $this->timer) {
return true;
} else {
return false;
}
}
public function setEnd($isEnd) {
$this->isEnd = $isEnd;
}
public function getEnd() {
return $this->isEnd;
}
}
/**
* 模擬阻塞的協(xié)程1
*
*/
function taskObject1() {
$timer = new timer(1,function($value,timer $timer) {
if($value >= 5) {
$timer->setEnd(true);
}
echo '<br>'.'A '.$value;
});
$tid = (yield getTaskId());
while (true) {
if($timer->getEnd() == true) {
break;
}
yield $timer->run();
}
}
/**
* 模擬阻塞的協(xié)程2
*
*/
function taskObject2() {
$timer = new timer(2,function($value,timer $timer) {
if($value >= 3) {
$timer->setEnd(true);
}
echo '<br>'.'B '.$value;
});
$tid = (yield getTaskId());
while (true) {
if($timer->getEnd() == true) {
break;
}
yield $timer->run();
}
}
$scheduler = new Scheduler;
$scheduler->newTask(taskObject1());
$scheduler->newTask(taskObject2());
$scheduler->run();
PHP應(yīng)用以上實(shí)現(xiàn)的是:
PHP應(yīng)用思考:
PHP應(yīng)用我為什么要做以上這件事情呢?因?yàn)槲野l(fā)現(xiàn)協(xié)程實(shí)現(xiàn)雖然很強(qiáng)大也很有意思,能讓多任務(wù)并行,但是我在其中一個(gè)任務(wù)里調(diào)用系統(tǒng)函數(shù) sleep()
的時(shí)候,阻塞任務(wù)會(huì)阻止協(xié)程切換,其實(shí)從協(xié)程的實(shí)現(xiàn)原理上來(lái)書也是這么回事.
PHP應(yīng)用那么,我也就想模擬協(xié)程阻塞,但是不產(chǎn)生阻塞看是否可行.PHP本身只提供了生成器為協(xié)程調(diào)用提供了支撐,如果不依賴擴(kuò)展,沒(méi)有提供多線程的程序?qū)崿F(xiàn)方式,沒(méi)有java那么強(qiáng)大,可以開(kāi)子線程進(jìn)行實(shí)現(xiàn).
PHP應(yīng)用我印象中java的子線程是獨(dú)立執(zhí)行且不會(huì)相互阻塞的,所以我在想,PHP既然可以實(shí)現(xiàn)類似于多線程這樣的機(jī)制,那么能不能實(shí)現(xiàn)調(diào)用過(guò)程中非阻塞呢?
PHP應(yīng)用經(jīng)過(guò)這樣一個(gè)實(shí)現(xiàn)和思考,一開(kāi)始是陷入了一個(gè)誤區(qū)的,是由于PHP原生函數(shù) sleep()
阻塞造成的思維誤區(qū),那就是認(rèn)為要想真正實(shí)現(xiàn)非阻塞或者說(shuō)實(shí)現(xiàn)異步的話,是必須依賴于語(yǔ)言底層的.
PHP應(yīng)用后來(lái),我想明白了一個(gè)道理,既然某個(gè)方法或者函數(shù)在執(zhí)行過(guò)程中,會(huì)產(chǎn)生阻塞,那么把當(dāng)前這個(gè)方法換成自定義的,做成非阻塞(相對(duì)于整個(gè)協(xié)程調(diào)度來(lái)說(shuō))不就行了嗎?比如上面的定時(shí)執(zhí)行我自己實(shí)現(xiàn)了一個(gè).
PHP應(yīng)用而另一方面,協(xié)程調(diào)度本身的目的也是為了把任務(wù)執(zhí)行過(guò)程切成盡量小片,從而快速切換執(zhí)行,達(dá)到并行的目的.從這方面來(lái)看,協(xié)程應(yīng)該也算是一種程序設(shè)計(jì)思想.
PHP應(yīng)用以下是一個(gè)程序切成盡量小片執(zhí)行的例子:
PHP應(yīng)用
// 一個(gè)簡(jiǎn)單的例子
<?php
function xrange($start, $end, $step = 1) {
for ($i = $start; $i <= $end; $i += $step) {
yield $i;
}
}
foreach (xrange(1, 1000000) as $num) {
echo $num, "\n";
}
PHP應(yīng)用這個(gè)例子是把原本用 range 生成一個(gè)很大的整型數(shù)組的方式切換為分片執(zhí)行,也就是說(shuō)在遍歷的時(shí)候再去取到指定的值,從代碼上來(lái)看,內(nèi)存消耗相對(duì)于之前來(lái)說(shuō)就非常小了.
PHP應(yīng)用總結(jié)
PHP應(yīng)用以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)維易PHP的支持.
轉(zhuǎn)載請(qǐng)注明本頁(yè)網(wǎng)址:
http://www.fzlkiss.com/jiaocheng/331.html