zoukankan      html  css  js  c++  java
  • PHP实现协同程序

    于server其中编程。为了实现异步。通常情况下,需要回调。比例如下面的代码

    function send($value) {
        $data = process($value);
        onReceive($data);
    }
    
    function onReceive($recv_value) {
        var_dump($recv_value);
    }
    
    function process($value) {
        return $value+1;
    }
    
    $send_value = 1;
    send($send_value);



    
    
    
    实现的东西非常easy,事实上就是将send_value发送到远端,远端server对其进行加一操作后,发送回来,于是在onReceive中我们能够得到远端server的返回值recv_value。
    


    可是这种代码就会看上去比較支离破碎,尤其是在process其中再次进行远程过程调用的时候。会变得更加难以开发和维护。协程就是为了解决这种问题,使得异步的代码看起来同步化。


    以下就是使用php的yield完毕代码调度的演示样例(假设想看懂这段代码。须要首先了解一下php 5.5的新特性generator和yield)

    框架代码例如以下:

    class CCoroutine {
    
        /**
         *
         * @var Generator 
         */
        public $coroutine;
    
        /**
         *
         * @var miexed null or CCoroutine
         */
        public $father;
    
        public function __construct($coroutine, $father = null) {
            $this->coroutine = $coroutine;
            $this->father = $father;
        }
    
    }
    
    class AsyncTask {
        
        public $data;
    
        public function __construct($data) {
            $this->data = $data;
        }
    
    }
    
    abstract class CoroutineScheduler {
        
        protected $coroutine = NULL;
    
        abstract function send_and_receive($value);
    
        public function run($data) {
            $co = $this->send_and_receive($data);
            $ccoroutine = new CCoroutine($co);
            $this->schedule($ccoroutine);
        }
    
        protected function schedule($ccoroutine) {
            $task = $ccoroutine->coroutine->current();
            //假设当前值为空,表示这个$ccoroutine应该已经结束了
            if (is_null($task)) {
                if (is_null($ccoroutine->father)) {
                //已经彻底调度结束了--通常是onRecieve方法执行到最后一步了
                    return;
                } else {
                //注意,假设执行到这个分支,则表示子生成器没有给父生成器传数据
                //子生成器可能是通过引用传递来改变父生成器的变量值
                //所以这个时候仅仅要调度父生成器就能够了
                    $ccoroutine->father->coroutine->next();
                    $father = $ccoroutine->father;
                    $this->schedule($father);
                    unset($ccoroutine);
                }
            } else {
                if (is_object($task) && $task instanceof AsyncTask) {
                    //当task是异步数据请求的时候,開始处理socket而且将进程熄火在这里
                    $this->dealTask($task, $ccoroutine);
                } elseif (is_object($task) && $task instanceof Generator) {
                    //当task是生成器时,表示当前生成器又有了子生成器的调用
                    $newcc = new CCoroutine($task, $ccoroutine);
                    $this->schedule($newcc);
                } elseif ($ccoroutine->father != null) {
                    //注意,假设执行到这个分支,则表示在子的生成器里调用了yield $str;这种写法
                    //我们规定这种写法是在给父生成器传数据,所以当前生成器就会终止调用了转而去调度父生成器
                    $ccoroutine->father->coroutine->send($task);
                    $father = $ccoroutine->father;
                    $this->schedule($father);
                    unset($ccoroutine);
                }
            }
        }
    
        protected function dealTask($task, $ccoroutine) {
            $this->coroutine = $ccoroutine;
            $this->send($task->data);
        }
        
        public function send($value) {
            $data = $this->process($value);
            $this->onReceive($data);
        }
    
        public function process($value) {
            return $value+1;
        }
        
        protected function onReceive($data) {
            $this->coroutine->coroutine->send($data);
            $this->schedule($this->coroutine);
        }
    
    }


    框架将 send, onReceive等函数所有都封装好了,使得调用方的代码看起来能够是同步的代码

    调用方代码例如以下:

    //1. 须要去实现CoroutineScheduler的send_and_receive函数,主要是为了在里面拿到返回值
    class Solution extends CoroutineScheduler {
    
        public function send_and_receive($data) {
            $result = (yield new AsyncTask($data));
            var_dump($result);
            
        }
    
    }
    
    //2. 在最外层去调用框架的代码。给出输入參数 $data
    $s = new Solution();
    $data = 1;
    $s->run($data);





    版权声明:本文博客原创文章,博客,未经同意,不得转载。

  • 相关阅读:
    Android Studio常用插件续
    Android Studio常用小技巧
    Repeater, DataList, 和GridView的区别及使用
    ListView在编辑状态下不能获取修改后的值,无法更新数据
    ASP.NET使用ListView数据绑定控件和DataPager实现数据分页显示(二)
    ASP.NET使用ListView数据绑定控件和DataPager实现数据分页显示(一)
    使用Sql分页方法给Repeater控件分页的方法
    使用生活实例理解Asp.net运行时
    .net4.0注册到IIS ,重新注册IIS ,iis注册
    关于.NET编译的目标平台(AnyCPU,x86,x64)
  • 原文地址:https://www.cnblogs.com/mfrbuaa/p/4719265.html
Copyright © 2011-2022 走看看