This article mainly introduces you to some understanding and thinking about coroutines and blocking in PHP. The article introduces it in detail through example code. It has certain reference learning value for everyone to learn or use PHP. Friends who need it Let’s learn with the editor below.
Preface
This article mainly introduces to you the understanding and thinking about coroutines and blocking in PHP, and shares it for your reference and study. Not much to say below, let’s take a look at the detailed introduction:
Process, thread, coroutine
About process , threads, and coroutines, there are very detailed and rich blogs or learning resources. I will not go into details here. I will briefly introduce these things here.
The process has its own independent heap and stack. It neither shares the heap nor the stack. The process is scheduled by the operating system.
Threads have their own independent stack and shared heap. The heap is shared but the stack is not shared. Threads are also scheduled by the operating system (standard threads are).
Coroutines share the heap like threads, but do not share stacks. Coroutines are scheduled explicitly by programmers in the code of the coroutine.
Coroutine implementation in PHP is based on yield
The fundamental implementation of yield is to generate Iterator class, and the iterator class is the implementation of the iterator interface:
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 ) // 序列化回调,抛出一个异常以表示生成器不能被序列化。 }
For the above analysis, please refer to the official PHP documentation.
http://php.net/manual/zh/clas...
And this detailed document:
http://www.jb51.net/ article/39424_all.htm
I will give an example based on the coroutine multi-task scheduling he implemented and talk about some of my thoughts on blocking.
Example of customizing simple scheduled execution tasks:
(This example must rely on the above coroutine scheduling code implemented by Brother Niao )
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();
The above implementation is:
Generate two tasks, execute them in parallel, and give each task the execution time Simulate a few seconds of blocking;
Let the coroutine switch smoothly, and the task blocking does not affect each other;
Thinking:
Why should I do the above thing? Because I found that although the coroutine implementation is very powerful and interesting, and can enable multi-tasking in parallel, when I call the system function sleep()
in one of the tasks, the blocking task will prevent the coroutine switching. In fact, This is also the case from the book's perspective on the implementation principles of coroutines.
Then, I also want to simulate coroutine blocking, but do not cause blocking to see if it is feasible. PHP itself only provides generators to provide support for coroutine calls. If it does not rely on extensions, it does not provide a multi-threaded program implementation method. It is not as powerful as Java and can be implemented by opening sub-threads.
I have the impression that Java's sub-threads execute independently and do not block each other, so I am thinking that since PHP can implement a mechanism similar to multi-threading, can it achieve non-blocking during the call process? ?
After such an implementation and thinking, I fell into a misunderstanding at the beginning. It was a misunderstanding caused by the blocking of the PHP native function sleep()
, that is, in order to truly achieve something special, Blocking or asynchronous implementation must depend on the underlying language.
Later, I figured out a truth. Since a certain method or function will block during execution, then replace the current method with a custom one and make it non-blocking (relative to the entire protocol). In terms of scheduling, wouldn’t it be enough? For example, I implemented the above scheduled execution myself.
On the other hand, the purpose of coroutine scheduling itself is to cut the task execution process into as small pieces as possible, so as to quickly switch execution and achieve parallelism. From this perspective, coroutine should also be regarded as a programming idea.
The following is an example of cutting a program into as small pieces as possible for execution:
##
// 一个简单的例子 <?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"; }This example is to use range The method of generating a large integer array is switched to shard execution, which means that the specified value is obtained during traversal. From the code point of view, the memory consumption is very small compared to before.
The above is the detailed content of An introduction to coroutines and blocking in PHP. For more information, please follow other related articles on the PHP Chinese website!

php把负数转为正整数的方法:1、使用abs()函数将负数转为正数,使用intval()函数对正数取整,转为正整数,语法“intval(abs($number))”;2、利用“~”位运算符将负数取反加一,语法“~$number + 1”。

实现方法:1、使用“sleep(延迟秒数)”语句,可延迟执行函数若干秒;2、使用“time_nanosleep(延迟秒数,延迟纳秒数)”语句,可延迟执行函数若干秒和纳秒;3、使用“time_sleep_until(time()+7)”语句。

php除以100保留两位小数的方法:1、利用“/”运算符进行除法运算,语法“数值 / 100”;2、使用“number_format(除法结果, 2)”或“sprintf("%.2f",除法结果)”语句进行四舍五入的处理值,并保留两位小数。

php字符串有下标。在PHP中,下标不仅可以应用于数组和对象,还可应用于字符串,利用字符串的下标和中括号“[]”可以访问指定索引位置的字符,并对该字符进行读写,语法“字符串名[下标值]”;字符串的下标值(索引值)只能是整数类型,起始值为0。

判断方法:1、使用“strtotime("年-月-日")”语句将给定的年月日转换为时间戳格式;2、用“date("z",时间戳)+1”语句计算指定时间戳是一年的第几天。date()返回的天数是从0开始计算的,因此真实天数需要在此基础上加1。

在php中,可以使用substr()函数来读取字符串后几个字符,只需要将该函数的第二个参数设置为负值,第三个参数省略即可;语法为“substr(字符串,-n)”,表示读取从字符串结尾处向前数第n个字符开始,直到字符串结尾的全部字符。

方法:1、用“str_replace(" ","其他字符",$str)”语句,可将nbsp符替换为其他字符;2、用“preg_replace("/(\s|\ \;||\xc2\xa0)/","其他字符",$str)”语句。

php判断有没有小数点的方法:1、使用“strpos(数字字符串,'.')”语法,如果返回小数点在字符串中第一次出现的位置,则有小数点;2、使用“strrpos(数字字符串,'.')”语句,如果返回小数点在字符串中最后一次出现的位置,则有。


Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

PhpStorm Mac version
The latest (2018.2.1) professional PHP integrated development tool

VSCode Windows 64-bit Download
A free and powerful IDE editor launched by Microsoft

WebStorm Mac version
Useful JavaScript development tools

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)
