成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

關于PHP協程與阻塞的思考

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

關于PHP協程與阻塞的思考

進程、線程、協程

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

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

PHP中的協程實現基礎 yield

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

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

 

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

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

以及鳥哥翻譯的這篇詳細文檔:

http://www.laruence.com/2015/...

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

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

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

  1. class timer { 
  2.     private $start = 0; // 定時開始時間 
  3.     private $timer; // 間隔的時間差,單位秒 
  4.     private $value = 0; // 產生的結果值 
  5.     private $callback; // 異步回調 
  6.     private $isEnd = false; // 當前定時器任務是否結束 
  7.     public function __construct($timer,callable $callback) 
  8.     { 
  9.         $this->start = time(); 
  10.         $this->timer = $timer; 
  11.         $this->callback = $callback; 
  12.     } 
  13.     public function run() { 
  14.         if($this->valid()) { 
  15.             $callback = $this->callback; 
  16.             $callback($this->value ++,$this); 
  17.             $this->start = time(); 
  18.         } 
  19.     } 
  20.     /** 
  21.      * 定時執行檢查 
  22.      */ 
  23.     public function valid() { 
  24.         $end = time(); 
  25.         if($end - $this->start >= $this->timer) { 
  26.             return true
  27.         } else { 
  28.             return false
  29.         } 
  30.     } 
  31.     public function setEnd($isEnd) { 
  32.         $this->isEnd = $isEnd; 
  33.     } 
  34.     public function getEnd() { 
  35.         return $this->isEnd; 
  36.     } 
  37.  
  38. /** 
  39.  * 模擬阻塞的協程1 
  40.  * 
  41.  */ 
  42. function taskObject1() { 
  43.     $timer = new timer(1,function($value,timer $timer) { 
  44.         if($value >= 5) { 
  45.             $timer->setEnd(true); 
  46.         } 
  47.         echo '<br>'.'A '.$value; 
  48.     }); 
  49.     $tid = (yield getTaskId()); 
  50.     while (true) { 
  51.         if($timer->getEnd() == true) { 
  52.             break; 
  53.         } 
  54.         yield $timer->run(); 
  55.     } 
  56. /** 
  57.  * 模擬阻塞的協程2 
  58.  * 
  59.  */ 
  60. function taskObject2() { 
  61.     $timer = new timer(2,function($value,timer $timer) { 
  62.         if($value >= 3) { 
  63.             $timer->setEnd(true); 
  64.         } 
  65.         echo '<br>'.'B '.$value; 
  66.     }); 
  67.     $tid = (yield getTaskId()); 
  68.     while (true) { 
  69.         if($timer->getEnd() == true) { 
  70.             break; 
  71.         } 
  72.         yield $timer->run(); 
  73.     } 
  74. $scheduler = new Scheduler; 
  75. $scheduler->newTask(taskObject1()); 
  76. $scheduler->newTask(taskObject2()); 
  77. $scheduler->run(); 

 

以上實現的是:

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

思考:

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

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

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

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

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

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

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

  1. // 一個簡單的例子 
  2. <?php 
  3. function xrange($start, $end, $step = 1) { 
  4.     for ($i = $start; $i <= $end; $i += $step) { 
  5.         yield $i; 
  6.     } 
  7.   
  8. foreach (xrange(1, 1000000) as $num) { 
  9.     echo $num, "\n"

 

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

責任編輯:龐桂玉 來源: segmentfault
相關推薦

2021-09-16 09:59:13

PythonJavaScript代碼

2021-06-15 07:10:14

JavaScript異步編程

2023-11-17 11:36:59

協程纖程操作系統

2023-12-24 12:56:36

協程

2016-10-28 17:39:47

phpgolangcoroutine

2017-05-02 11:38:00

PHP協程實現過程

2025-01-26 00:00:15

PHP協程控制權

2018-12-04 14:00:41

協程編程模式PHP

2025-06-26 04:10:00

2022-04-19 20:39:03

協程多進程

2023-10-12 09:46:00

并發模型線程

2024-02-05 09:06:25

Python協程Asyncio庫

2025-05-16 08:21:45

2024-06-27 07:56:49

2020-05-19 08:52:31

APP滲透測試終端安全

2023-10-24 19:37:34

協程Java

2021-12-09 06:41:56

Python協程多并發

2025-02-08 09:13:40

2017-09-22 16:08:16

Python協程編程

2025-06-03 00:00:02

Go協程鎖機制
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩at | 91爱爱·com | 国产精品一区二区久久久久 | 久久国产精品久久 | 中文字幕在线一区二区三区 | 欧美精品福利 | 一区在线观看视频 | 国产在线二区 | 欧美亚洲在线 | 亚洲一区二区三区四区五区中文 | 日韩视频在线免费观看 | 国产一区二区三区在线 | 一级免费a| 国产成人精品一区二区 | 亚洲色图在线观看 | 中文字幕一区二区三区在线观看 | 精品伊人| 欧美福利视频一区 | 国产综合精品 | 精品一区二区三区中文字幕 | 日本黄色大片免费看 | 一区二区三区免费观看 | 国产精品网址 | 91精品国产91久久久 | 精品免费视频 | 在线观看中文字幕亚洲 | 欧美精品在线一区 | 国产色片| 天堂免费看片 | 日韩欧美国产成人一区二区 | 国产女人与拘做受视频 | 亚洲第一区国产精品 | 日韩第一页 | 在线观看中文字幕 | 人人干人人舔 | 久久久久久国产精品mv | 一区二区三区不卡视频 | 日本免费在线 | 激情毛片 | 久草网在线视频 | 欧美精品三区 |