成人精品一区二区三区中文字幕-成人精品一区二区三区-成人精品一级毛片-成人精品亚洲-日本在线视频一区二区-日本在线视频免费

導航首頁 ? 技術教程 ? 關于PHP中協程和阻塞的一些理解與思考
全站頭部文字 我要出現在這里
關于PHP中協程和阻塞的一些理解與思考 721 2023-12-08   

前言

本文主要給大家介紹了關于PHP中協程和阻塞的理解與思考,分享出來供大家參考學習,下面話不多說了,來一起看看詳細的介紹:

進程、線程、協程

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

進程擁有自己獨立的堆和棧,既不共享堆,亦不共享棧,進程由操作系統調度。 線程擁有自己獨立的棧和共享的堆,共享堆,不共享棧,線程亦由操作系統調度(標準線程是的)。 協程和線程一樣共享堆,不共享棧,協程由程序員在協程的代碼里顯示調度。

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 ) // 序列化回調,拋出一個異常以表示生成器不能被序列化。
}

以上解析可以參考PHP官方文檔。

http://php.net/manual/zh/clas...

以及這篇詳細文檔:

http://www.gimoo.net/article/39424_all.htm

我就以他實現的協程多任務調度為基礎做一下例子說明并說一下關于我在阻塞方面所做的一些思考。

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

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

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();

以上實現的是:

產生兩個任務,并行執行,并且給每個任務在執行的時候模擬幾秒鐘的阻塞; 讓協程切換的時候能順利切換,其中的任務阻塞不相互影響;

思考:

我為什么要做以上這件事情呢?因為我發現協程實現雖然很強大也很有意思,能讓多任務并行,但是我在其中一個任務里調用系統函數 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 生成一個很大的整型數組的方式切換為分片執行,也就是說在遍歷的時候再去取到指定的值,從代碼上來看,內存消耗相對于之前來說就非常小了。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對綠夏網的支持。


主站蜘蛛池模板: 画皮电影| 红灯停绿灯行电影观看| 湖南卫视节目表今天| 狗年电影| 《爱的温暖》电影在线观看| 男女视频在线播放| 五年级下册语文第七单元口语交际| 索尼克音爆| 玛丽亚小泽| 朱莉安妮全集高清免费| 超级飞侠 第四季 动漫| av网址大全| 宝悦| 四月一日| alura jenson movies| 花宵道中1| 夜之女王 电影| 湖北卫视在线直播| 4人免费剧本及答案| 在线观看高清电影| k总直播间| 侠侣探案| 韩国一级免费| 铁探粤语版在线观看| 德川女刑罚绘卷| 母亲电影完整版韩国| 爱爱免费视频观看| 金靖星座| 老闺蜜电视剧免费观看完整版高清| 忘却的旋律| 痛风能吃豆腐吗| 大决战全部演员表介绍图片| department什么意思| 久草电影| 我的女老师| 5.25心理健康日主题班会ppt| 国产电影网站| 永远的乳房 电影| 蓝眼泪简谱| 王若晰的个人资料| 血色誓言演员表|

!!!站長長期在線接!!!

網站、小程序:定制開發/二次開發/仿制開發等

各種疑難雜癥解決/定制接口/定制采集等

站長微信:lxwl520520

站長QQ:1737366103