前言
本文主要給大家介紹了關于PHP中協(xié)程和阻塞的理解與思考,分享出來供大家參考學習,下面話不多說了,來一起看看詳細的介紹:
進程、線程、協(xié)程
關于進程、線程、協(xié)程,有非常詳細和豐富的博客或者學習資源,我不在此做贅述,我大致在此介紹一下這幾個東西。
- 進程擁有自己獨立的堆和棧,既不共享堆,亦不共享棧,進程由操作系統(tǒng)調度。
- 線程擁有自己獨立的棧和共享的堆,共享堆,不共享棧,線程亦由操作系統(tǒng)調度(標準線程是的)。
- 協(xié)程和線程一樣共享堆,不共享棧,協(xié)程由程序員在協(xié)程的代碼里顯示調度。
PHP中的協(xié)程實現(xiàn)基礎 yield
yield的根本實現(xiàn)是生成器類,而迭代器類是迭代器接口的實現(xiàn):
Generator implements Iterator { public mixed current ( void ) // 返回當前產生的值 public mixed key ( void ) // 返回當前產生的鍵 public void next ( void ) // 生成器繼續(xù)執(zhí)行 public void rewind ( void ) // 重置迭代器,如果迭代已經(jīng)開始了,這里會拋出一個異常。 // renwind的執(zhí)行將會導致第一個yield被執(zhí)行, 并且忽略了他的返回值. public mixed send ( mixed $value ) // 向生成器中傳入一個值,并且當做 yield 表達式的結果,然后繼續(xù)執(zhí)行生成器。如果當這個方法被調用時,生成器 // 不在 yield 表達式,那么在傳入值之前,它會先運行到第一個 yield 表達式。 public void throw ( Exception $exception ) // 向生成器中拋入一個異常 public bool valid ( void ) // 檢查迭代器是否被關閉 public void __wakeup ( void ) // 序列化回調,拋出一個異常以表示生成器不能被序列化。 }
以上解析可以參考PHP官方文檔。
http://php.net/manual/zh/clas...
以及這篇詳細文檔:
http://www.jb51.net/article/39424_all.htm
我就以他實現(xiàn)的協(xié)程多任務調度為基礎做一下例子說明并說一下關于我在阻塞方面所做的一些思考。
自定義簡單定時執(zhí)行任務示例:
(此例子必須依賴于以上鳥哥實現(xiàn)的協(xié)程調度代碼)
class timer { private $start = 0; // 定時開始時間 private $timer; // 間隔的時間差,單位秒 private $value = 0; // 產生的結果值 private $callback; // 異步回調 private $isEnd = false; // 當前定時器任務是否結束 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(); } } /** * 定時執(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();
以上實現(xiàn)的是:
- 產生兩個任務,并行執(zhí)行,并且給每個任務在執(zhí)行的時候模擬幾秒鐘的阻塞;
- 讓協(xié)程切換的時候能順利切換,其中的任務阻塞不相互影響;
思考:
我為什么要做以上這件事情呢?因為我發(fā)現(xiàn)協(xié)程實現(xiàn)雖然很強大也很有意思,能讓多任務并行,但是我在其中一個任務里調用系統(tǒng)函數(shù) sleep()
的時候,阻塞任務會阻止協(xié)程切換,其實從協(xié)程的實現(xiàn)原理上來書也是這么回事。
那么,我也就想模擬協(xié)程阻塞,但是不產生阻塞看是否可行。PHP本身只提供了生成器為協(xié)程調用提供了支撐,如果不依賴擴展,沒有提供多線程的程序實現(xiàn)方式,沒有java那么強大,可以開子線程進行實現(xiàn)。
【網(wǎng)站聲明】本站除付費源碼經(jīng)過測試外,其他素材未做測試,不保證完整性,網(wǎng)站上部分源碼僅限學習交流,請勿用于商業(yè)用途。如損害你的權益請聯(lián)系客服QQ:2655101040 給予處理,謝謝支持。