PHP协程

1.什么是协程
先搞清楚,什么是协程。

你可能已经听过『进程』和『线程』这两个概念。

进程就是二进制可执行文件在计算机内存里的一个运行实例,就好比你的.exe文件是个类,进程就是new出来的那个实例。

进程是计算机系统进行资源分配和调度的基本单位(调度单位这里别纠结线程进程的),每个CPU下同一时刻只能处理一个进程。

所谓的并行,只不过是看起来并行,CPU事实上在用很快的速度切换不同的进程。

进程的切换需要进行系统调用,CPU要保存当前进程的各个信息,同时还会使CPUCache被废掉。

所以进程切换不到非不得已就不做。

那么怎么实现『进程切换不到非不得已就不做』呢?

首先进程被切换的条件是:进程执行完毕、分配给进程的CPU时间片结束,系统发生中断需要处理,或者进程等待必要的资源(进程阻塞)等。你想下,前面几种情况自然没有什么话可说,但是如果是在阻塞等待,是不是就浪费了。

其实阻塞的话我们的程序还有其他可执行的地方可以执行,不一定要傻傻的等!所以就有了线程。线程简单理解就是一个『微进程』,专门跑一个函数(逻辑流)。

所以我们就可以在编写程序的过程中将可以同时运行的函数用线程来体现了。

线程有两种类型,一种是由内核来管理和调度。

我们说,只要涉及需要内核参与管理调度的,代价都是很大的。这种线程其实也就解决了当一个进程中,某个正在执行的线程遇到阻塞,我们可以调度另外一个可运行的线程来跑,但是还是在同一个进程里,所以没有了进程切换。

还有另外一种线程,他的调度是由程序员自己写程序来管理的,对内核来说不可见。这种线程叫做『用户空间线程』。
协程可以理解就是一种用户空间线程。

协程,有几个特点:

协同,因为是由程序员自己写的调度策略,其通过协作而不是抢占来进行切换

在用户态完成创建,切换和销毁

⚠️ 从编程角度上看,协程的思想本质上就是控制流的主动让出(yield)和恢复(resume)机制

迭代器经常用来实现协程
2.迭代器的实现原理:

可迭代对象

PHP5提供了一种定义对象的方法使其可以通过单元列表来遍历,例如用foreach语句。

你如果要实现一个可迭代对象,你就要实现Iterator接口:

class MyIterator extends Iterator {
    private $var = [];
    public function __construct(array $array = array())
    {
        if(is_array($array)){
            $this->var = $array;
        }
    }

    //重置索引游标的指向第一个元素
    public function rewind(){
        echo "rewinding\n";
        reset($this->var);
    }
    //返回当前索引游标指向的元素
    public function current()
    {
        $var = current($this->var);
        echo "current: $var\n";
        return $var;
    }
    //返回当前索引游标指向的元素的键名
    public function key(){
        $var = key($this->var);
        echo "key: $var\n";
        return $var;
    }

    //移动当前索引游标指向下一元素
    public function next() {
        $var = next($this->var);
        echo "next: $var\n";
        return $var;
    }
    public function valid() {  //如果执行valid返回false,则循环就此结束
        $var = $this->current() !== false;
        echo "valid: {$var}\n";
        return $var;
    }

}

调用迭代器示例:

       $values = ['php'=>'wudy','java'=>'timo','python'=>'peter'];
        $it = new MyIterator($values);
        foreach ($it as $key => $value) {
            print "$key: $value\n";
        }

输出结果:

rewinding
current: wudy
valid: 1
current: wudy
key: php
php: wudy
next: timo
current: timo
valid: 1
current: timo
key: java
java: timo
next: peter
current: peter
valid: 1
current: peter
key: python
python: peter
next: 
current: 
valid: 

可以看出,迭代器对象会首先调用rewind方法,其调用顺序为 rewind->current->valid->current->key->next->...,依次循环 , 直到valid返回false,则终止继续遍历

生成器

可以说之前为了拥有一个能够被foreach遍历的对象,你不得不去实现一堆的方法,yield关键字就是为了简化这个过程。
生成器提供了一种更容易的方法来实现简单的对象迭代,相比较定义类实现Iterator接口的方式,性能开销和复杂性大大降低。

function xrange($start,$end,$step=1){
        for($i=$start; $i<=$end; $i+=$step){
            yield $i;
        }
   }
//调用
foreach ($this->xrange(1, 1000000) as $num){
            echo $num,"\n";
        }

记住,一个函数中如果用了yield,他就是一个生成器,直接调用他是没有用的,不能等同于一个函数那样去执行!

PHP协程

前面介绍协程的时候说了,协程需要程序员自己去编写调度机制,下面我们来看这个机制怎么写。

0)生成器正确使用

既然生成器不能像函数一样直接调用,那么怎么才能调用呢?

方法如下:

foreach他

send($value)

current / next...

1)Task实现

Task就是一个任务的抽象,刚刚我们说了协程就是用户空间协程,线程可以理解就是跑一个函数。
所以Task的构造函数中就是接收一个闭包函数,我们命名为coroutine。

//Task任务类
use Generator;
class Task{
   protected $taskId;
   protected $coroutine;
   protected $beforeFirstYield=true;
   protected $sendValue;
   /**
    * Task constructor.
    * @param $taskId
    * @param Generator $coroutine
    */
   public function __construct($taskId, Generator $coroutine)
   {
       $this->taskId = $taskId;
       $this->coroutine = $coroutine;
   }

   /**
    * 获取当前的Task的ID
    * @return mixed
    */
   public function getTaskId(){
       return $this->taskId;
   }

   /**
    * 判断Task执行完毕了没有
    * @return bool
    */
   public function isFinished(){
       return !$this->coroutine->valid();
   }

   /**
    * 设置下次要传给协程的值,比如 $id = (yield $xxxx),这个值就给了$id了
    * @param $value
    */
   public function setSendvalue($value){
       $this->sendValue = $value;
   }

   /**
    * 运行任务
    * @return mixed
    */
   public function run(){
       // 这里要注意,生成器的开始会reset,所以第一个值要用current获取
       if($this->beforeFirstYield){
           $this->beforeFirstYield = false;
           return $this->coroutine->current();
       }else{
           $retval = $this->coroutine->send($this->sendValue);
           $this->sendValue = null;
           return $retval;
       }
   }
}

2)Scheduler实现
接下来就是Scheduler这个重点核心部分,他扮演着调度员的角色

use SplQueue;
class Scheduler{
    protected $taskQueue;
    protected $tid=0;
    public function __construct()
    {
        /*
        *  原理就是维护了一个队列,
        *  前面说过,从编程角度上看,协程的思想本质上就是控制流的主动让出(yield)和恢复(resume)机制
        * */
        $this->taskQueue = new SplQueue();
    }

    /**
     * 增加一个任务
     * @param Generator $task
     * @return int
     */
    public function addTask(\Generator $task){
        $tid = $this->tid;
        $task = new Task($tid,$task);
        $this->taskQueue->enqueue($task);
        $this->tid;
        return $tid;
    }

    /**
     * 把任务进入队列
     * @param Task $task
     */
    public function schedule(Task $task){
        $this->taskQueue->enqueue($task);
    }

    /**
     * 运行调度器
     */
    public function run(){
        while(!$this->taskQueue->isEmpty()){
            //任务出列
            $task = $this->taskQueue->dequeue();
            $res = $task->run();  // 运行任务直到 yield
            if(!$task->isFinished()){
                $this->schedule($task);// 任务如果还没完全执行完毕,入队等下次执行
            }
        }
    }
}

这样我们基本就实现了一个协程调度器。

你可以使用下面的代码来测试:

       $scheduler = new Scheduler();  // 实例化一个调度器
        $scheduler->addTask($this->task1());  // 添加不同的闭包函数作为任务
        $scheduler->addTask($this->task2());
        $scheduler->run();

输出结果:

This is task 1 iteration 1.
This is task 2 iteration 1.
This is task 1 iteration 2.
This is task 2 iteration 2.
This is task 1 iteration 3.
This is task 2 iteration 3.
This is task 1 iteration 4.
This is task 2 iteration 4.
This is task 1 iteration 5.
This is task 2 iteration 5.
This is task 1 iteration 6.
This is task 1 iteration 7.
This is task 1 iteration 8.
This is task 1 iteration 9.
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,165评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,720评论 1 298
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,849评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,245评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,596评论 3 288
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,747评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,977评论 2 315
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,708评论 0 204
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,448评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,657评论 2 249
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,141评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,493评论 3 258
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,153评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,108评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,890评论 0 198
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,799评论 2 277
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,685评论 2 272

推荐阅读更多精彩内容