前言
第一部分: 半协程调度器
- 统一生成器接口
- 生成器迭代
- 生成器返回值
- 生成器委托
- 改写 return
- 抽象异步模型
- 引入异常处理
- 异常: 嵌套任务透传
- 异常: 传递流程
- 异常: 重新进行 CPS 变换
- 异常: 重新加入 Async
- Syscall 与 Context
- 调度器: 里程碑
- spawn
- callcc
- race 与 timeout
- all 与 parallel
- channel 与协程间通信
- 无缓存 channel
- 缓存 channel
- channel 演示
- FutureTask 与 fork
第二部分: Koa
- 穿越地心之旅
- 洋葱圈模型
- rightReduce与中间件compose
- Koa::Application
- Koa::Context
- Koa::Request
- Koa::Response
- Koa - HelloWorld
- Middleware Interface
- Middleware: 全局异常处理
- Middleware: Router
- Middleware: 请求超时
- 一个综合示例
附录
参考
文章来源于网络收集而来,版权归原创者所有,如有侵权请及时联系!
无缓存 channel
无缓存channel
我们首先实现无缓存的Channel:
<?php
class Channel
{
// 因为同一个channel可能有多个接收者,使用队列实现,保证调度均衡
// 队列内保存的是被阻塞的接收者协程的控制流,即call/cc的参数,我们模拟的continuation
public $recvQ;
// 发送者队列逻辑相同
public $sendQ;
public function __construct()
{
$this->recvQ = new \SplQueue();
$this->sendQ = new \SplQueue();
}
public function send($val)
{
return callcc(function($cc) use($val) {
if ($this->recvQ->isEmpty()) {
// 当chan没有接收者,发送者协程挂起(将$cc入列,不调用$cc回送数据)
$this->sendQ->enqueue([$cc, $val]);
} else {
// 当chan对端有接收者,将挂起接收者协程出列,
// 调用接收者$recvCc发送数据,运行接收者协程后继代码
// 执行完毕或者遇到Async挂起,$recvCc()调用返回,
// 调用$cc(),控制流回到发送者协程
$recvCc = $this->recvQ->dequeue();
$recvCc($val, null);
$cc(null, null);
}
});
}
public function recv()
{
return callcc(function($cc) {
if ($this->sendQ->isEmpty()) {
// 当chan没有发送者,接收者协程挂起(将$cc入列)
$this->recvQ->enqueue($cc);
} else {
// 当chan对端有发送者,将挂起发送者协程与待发送数据出列
// 调用发送者$sendCc发送数据,运行发送者协程后继代码
// 执行完毕或者遇到Async挂起,$sendCc()调用返回,
// 调用$cc(),控制流回到接收者协程
list($sendCc, $val) = $this->sendQ->dequeue();
$sendCc(null, null);
$cc($val, null);
}
});
}
}
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论