中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

PHP協程與阻塞舉例分析

發布時間:2021-11-19 17:10:40 來源:億速云 閱讀:134 作者:iii 欄目:編程語言

本篇內容介紹了“PHP協程與阻塞舉例分析”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

進程、線程、協程

關于進程、線程、協程,有非常詳細和豐富的博客或者學習資源,我不在此做贅述,我大致在此介紹一下這幾個東西。

  1. 進程擁有自己獨立的堆和棧,既不共享堆,亦不共享棧,進程由操作系統調度。

  2. 線程擁有自己獨立的棧和共享的堆,共享堆,不共享棧,線程亦由操作系統調度(標準線程是的)。

  3. 協程和線程一樣共享堆,不共享棧,協程由程序員在協程的代碼里顯示調度。

PHP中的協程實現基礎 yield

yield的根本實現是生成器類,而迭代器類是迭代器接口的實現:

Generator implements Iterator {     public mixed current ( void ) // 返回當前產生的值     public mixed key ( void ) // 返回當前產生的鍵     public void next ( void ) // 生成器繼續執行     public void rewind ( void ) // 重置迭代器,如果迭代已經開始了,這里會拋出一個異常。                                              // renwind的執行將會導致***個yield被執行, 并且忽略了他的返回值.     public mixed send ( mixed $value ) // 向生成器中傳入一個值,并且當做 yield 表達式的結果,然后繼續執行生成器。如果當這個方法被調用時,生成器                                                // 不在 yield 表達式,那么在傳入值之前,它會先運行到***個 yield 表達式。     public void throw ( Exception $exception ) // 向生成器中拋入一個異常     public bool valid ( void ) // 檢查迭代器是否被關閉     public void __wakeup ( void ) // 序列化回調,拋出一個異常以表示生成器不能被序列化。 }

自定義簡單定時執行任務示例:

(此例子必須依賴于以上鳥哥實現的協程調度代碼)

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();         }     }     /**      * 定時執行檢查      */     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;     } }  /**  * 模擬阻塞的協程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();     } } /**  * 模擬阻塞的協程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();

以上實現的是:

  1. 產生兩個任務,并行執行,并且給每個任務在執行的時候模擬幾秒鐘的阻塞;

  2. 讓協程切換的時候能順利切換,其中的任務阻塞不相互影響;

思考:

我為什么要做以上這件事情呢?因為我發現協程實現雖然很強大也很有意思,能讓多任務并行,但是我在其中一個任務里調用系統函數 sleep()  的時候,阻塞任務會阻止協程切換,其實從協程的實現原理上來書也是這么回事。

那么,我也就想模擬協程阻塞,但是不產生阻塞看是否可行。PHP本身只提供了生成器為協程調用提供了支撐,如果不依賴擴展,沒有提供多線程的程序實現方式,沒有java那么強大,可以開子線程進行實現。

我印象中java的子線程是獨立執行且不會相互阻塞的,所以我在想,PHP既然可以實現類似于多線程這樣的機制,那么能不能實現調用過程中非阻塞呢?

經過這樣一個實現和思考,一開始是陷入了一個誤區的,是由于PHP原生函數 sleep()  阻塞造成的思維誤區,那就是認為要想真正實現非阻塞或者說實現異步的話,是必須依賴于語言底層的。

后來,我想明白了一個道理,既然某個方法或者函數在執行過程中,會產生阻塞,那么把當前這個方法換成自定義的,做成非阻塞(相對于整個協程調度來說)不就行了嗎?比如上面的定時執行我自己實現了一個。

而另一方面,協程調度本身的目的也是為了把任務執行過程切成盡量小片,從而快速切換執行,達到并行的目的。從這方面來看,協程應該也算是一種程序設計思想。

以下是一個程序切成盡量小片執行的例子:

// 一個簡單的例子 <?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"; }

這個例子是把原本用 range  生成一個很大的整型數組的方式切換為分片執行,也就是說在遍歷的時候再去取到指定的值,從代碼上來看,內存消耗相對于之前來說就非常小了。

“PHP協程與阻塞舉例分析”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!

向AI問一下細節
推薦閱讀:
  1. lua 協程
  2. GO協程

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

php
AI

岱山县| 奎屯市| 沂源县| 庄浪县| 上虞市| 湖北省| 阳东县| 临高县| 拉萨市| 湖口县| 顺平县| 临城县| 夏河县| 东乡县| 武清区| 新和县| 长垣县| 三都| 隆安县| 江源县| 黄大仙区| 涡阳县| 富平县| 修水县| 静宁县| 平邑县| 汉源县| 宝坻区| 五大连池市| 册亨县| 隆德县| 盐池县| 平遥县| 安徽省| 杭州市| 夏邑县| 长乐市| 丰顺县| 将乐县| 屯留县| 廊坊市|