PHP下的异步尝试一:初识生成器
PHP下的异步尝试系列
- PHP下的异步尝试一:初识生成器
- PHP下的异步尝试二:初识协程
- PHP下的异步尝试三:协程的PHP版thunkify自动执行器
- PHP下的异步尝试四:PHP版的Promise
- [PHP下的异步尝试五:PHP版的Promise的继续完善]
生成器类
# http://php.net/manual/zh/class.generator.php
Generator implements Iterator {
/* Methods */
//获取迭代器当前值
public mixed current ( void )
//获取迭代器当前值
public mixed getReturn ( void )
//返回当前产生的键
public mixed key ( void )
//生成器从上一次yield处继续执行
public void next ( void )
//重置迭代器
public void rewind ( void )
//向生成器中传入一个值
public mixed send ( mixed $value )
//向生成器中抛入一个异常
public mixed throw ( Throwable $exception )
//检查迭代器是否被关闭
public bool valid ( void )
//迭代器序列化时执行的方法
public void __wakeup ( void )
}
生成生成器
尝试实例化类
$gen = new Generator();
# 我们发现不能直接手动实例化
# output
PHP Fatal error: Uncaught Error: The "Generator" class is reserved for internal use and cannot be manually instantiated in /web/www/sxx_admin3/src/cache/test/amphp/gen3.php:8
尝试function方式
function gen($max)
{
for ($i=0; $i<$max; $i++) {
yield $i;
}
}
$gen = gen(5);
# success
# 成功,我们只需要在普通函数方法里yield即可成了生成器
理解php的生成器
其实各语言都有生成器,比如python,go等
生成器迭代foreach
被代码将演示valid, getReturn
function gen($max)
{
for ($i=0; $i<$max; $i++) {
yield $i;
}
return $max;
}
$gen = gen(5);
foreach ($gen as $val) {
var_dump($val);
}
//如果已经迭代完成,获取返回值
// php7 支持
// valid 判断当前迭代器是否迭代完成
// getReturn 返回迭代器的返回值
if (version_compare(PHP_VERSION, '7.0.0') >= 0 && !$gen->valid()) {
var_dump($gen->getReturn());
}
带key值的生成器迭代foreach
迭代器返回值可以带key和value,类似
function gen($max)
{
for ($i=0; $i<$max; $i++) {
yield $i => $i+1;
}
return $max;
}
$gen = gen(5);
//var_dump($gen->key());
//var_dump($gen->current());
foreach ($gen as $key=>$val) {
var_dump($key . "=>" . $val);
}
# output
string(4) "0=>1"
string(4) "1=>2"
string(4) "2=>3"
string(4) "3=>4"
string(4) "4=>5"
生成器迭代手动迭代
本代码将演示rewind, next, send方法
function gen($max)
{
for ($i=0; $i<$max; $i++) {
// 此处的(yield $i)在php7以后版本可省略
$res = (yield $i);
var_dump($res);
}
return $max;
}
$gen = gen(10);
// 可不调用,隐式调用
// 如果迭代开始后不能再rewind(即使用了next或send后)
$gen->rewind();
// 打印获取到当前生成器的值
var_dump("1::" . $gen->current()); //output: string(4) "1::0"
// 下面2句代码执行,将返回错误
// $gen->next();
// $gen->rewind();
//继续执行,知道遇到下一个yield
$gen->next();
var_dump("2::" . $gen->current()); //output: string(4) "2::1"
$gen->next();
var_dump("3::" . $gen->current()); //output: string(4) "3::2"
// send传null值等同于调用next(本方法尝试来自python的迭代器,成功)
$gen->send(null);
var_dump("4::" . $gen->current()); //output: string(4) "4::3"
// send传值会也会继续执行
$gen->send(100);
var_dump("5::" . $gen->current()); //output: string(4) "5::4"
//如果已经迭代完成,获取返回值
// php7 支持
if (version_compare(PHP_VERSION, '7.0.0') >= 0 && !$gen->valid()) {
var_dump($gen->getReturn());
}
# output:
string(4) "1::0"
NULL
string(4) "2::1"
NULL
string(4) "3::2"
NULL
string(4) "4::3"
int(100)
string(4) "5::4"
# 我们先不去理会gen里var_dump输出的NULL或int(100)
# 我们先去理解每次next后current可以获取到当前yield的值即可
尝试理解send输出
function gen($max)
{
for ($i=0; $i<$max; $i++) {
$res = (yield $i);
var_dump($res);
}
return $max;
}
$gen = gen(10);
var_dump("1::" . $gen->current());
$gen->send(222);
var_dump("2::" . $gen->current());
$gen->send(333);
var_dump("3::" . $gen->current());
$gen->send(null);
var_dump("4::" . $gen->current());
# output:
string(4) "1::0"
int(222)
string(4) "2::1"
int(333)
string(4) "3::2"
int(444)
string(4) "4::3"
# send和next
# next() => current = yield值
# send(val) $rs = yield 表达式执行 = val; //send这样理解即可
# 在当前某个yield处时send,当前yield表达式处返回,如果没有变量接收,那么继续下一个yield处返回
$rs = (yield somethind_to_do(...) );
^ |-------------------|
| yield值
| |----------------------------|
| yield 表达式
yield表达式结果
# 执行顺序流程类似
$res = (yield 1); // <- var_dump("1::" . $gen->current()); 第一步到yield返回
var_dump($res); // <- $gen->send(222); 第二步send:222后,继续往下走$res=222 然后var_dump($res), 然后到了yield 2
$res = (yield 2); // <- var_dump("2::" . $gen->current()); 打印当前的值2
var_dump($res); // <- $gen->send(333); 第三步send:333后,继续往下走$res=333 然后var_dump($res), 然后到了yield 3
$res = (yield 3); // <- var_dump("3::" . $gen->current());
var_dump($res); // <- $gen->send(null); 第二步send:null后,继续往下走$res=null 然后var_dump($res), 然后到了yield 4
$res = (yield 4); // <- var_dump("4::" . $gen->current());
生成器throw抛出错误
# 内部定义异常并返回,外部接收
function gen() {
echo "Gen 开始\n";
yield new Exception('内部定义异常');
echo "Gen 结束\n";
}
$gen = gen();
try {
throw $gen->current();
} catch (\Exception $e) {
echo "外部捕获异常:" . $e->getMessage() . PHP_EOL;
}
$gen->send("123");
# output:
Gen 开始
外部捕获异常:内部定义异常
Gen 结束
# 内部接收send传入的异常,然后直接throw,外部接收
function gen() {
echo "Gen 开始\n";
throw (yield new Exception('内部定义异常'));
echo "Gen 结束\n";
}
$gen = gen();
try {
throw $gen->current();
} catch (\Exception $e) {
echo "外部捕获异常:" . $e->getMessage() . PHP_EOL;
}
try {
$gen->send(new \Exception("外部定义异常"));
} catch (\Exception $e) {
echo "外部捕获异常:" . $e->getMessage() . PHP_EOL;
}
# output
Gen 开始
外部捕获异常:内部定义异常
外部捕获异常:外部定义异常
#
function gen() {
echo "Gen 开始\n";
try {
yield new Exception('内部定义异常');
} catch (Exception $e) {
echo "内部捕获异常:" . "Exception: {$e->getMessage()}\n";
}
echo "Gen 结束\n";
}
$gen = gen();
try {
throw $gen->current();
} catch (\Exception $e) {
echo "外部捕获异常:" . $e->getMessage() . PHP_EOL;
}
// 注意这里是throw方法,相当于
// $gen->send(new \Exception("外部定义异常"));
// throw yield (yield接收 = new \Exception("外部定义异常"))
$gen->throw(new \Exception("外部定义异常"));
#output
Gen 开始
外部捕获异常:内部定义异常
内部捕获异常:Exception: 外部定义异常
Gen 结束
总结
初识我们只需要先理解next和send即可
next->让我们可以主动自动执行迭代器
send->可以让我们的迭代器实现双向通信,改变执行体流程顺序
后续我们会介绍使用场景和Co自动执行体等
文章更名记录
2018.09.20: [PHP下的生成器尝试一:初识PHP下的生成器] -> [PHP下的异步尝试一:初识生成器]
原文地址:https://segmentfault.com/a/1190000016444494
PHP下的异步尝试一:初识生成器的更多相关文章
- PHP下的异步尝试二:初识协程
PHP下的异步尝试系列 如果你还不太了解PHP下的生成器,你可以根据下面目录翻阅 PHP下的异步尝试一:初识生成器 PHP下的异步尝试二:初识协程 PHP下的异步尝试三:协程的PHP版thunkify ...
- PHP下的异步尝试四:PHP版的Promise
PHP下的异步尝试系列 如果你还不太了解PHP下的生成器和协程,你可以根据下面目录翻阅 PHP下的异步尝试一:初识生成器 PHP下的异步尝试二:初识协程 PHP下的异步尝试三:协程的PHP版thunk ...
- PHP下的异步尝试三:协程的PHP版thunkify自动执行器
PHP下的异步尝试系列 如果你还不太了解PHP下的生成器和协程,你可以根据下面目录翻阅 PHP下的异步尝试一:初识生成器 PHP下的异步尝试二:初识协程 PHP下的异步尝试三:协程的PHP版thunk ...
- 刚刚研究了下ipv6,尝试配置内网VPS的IPv6地址
刚刚研究了下ipv6,尝试配置内网VPS的IPv6地址是3台设备,分别是客户机Windows系统.核心交换机.PPPoE拨号的路由器 第一步:在PPPoE拨号的路由器上面查看ppp0拨号的地址 ifc ...
- PHP 命令行模式实战之cli+mysql 模拟队列批量发送邮件(在Linux环境下PHP 异步执行脚本发送事件通知消息实际案例)
源码地址:https://github.com/Tinywan/PHP_Experience 测试环境配置: 环境:Windows 7系统 .PHP7.0.Apache服务器 PHP框架:ThinkP ...
- 初探.net framework 下的异步多线程
初探.net framework 下的异步多线程 目录 1.多线程的出现条件 2.Thread和ThreadPool的相关Api及用法 3.Task和Parallel的相关Api及用法 4.Async ...
- 【神经网络与深度学习】【CUDA开发】caffe-windows win32下的编译尝试
[神经网络与深度学习][CUDA开发]caffe-windows win32下的编译尝试 标签:[神经网络与深度学习] [CUDA开发] 主要是在开发Qt的应用程序时,需要的是有一个使用的库文件也只是 ...
- javascript异步编程之generator(生成器函数)与asnyc/await语法糖
Generator 异步方案 相比于传统回调函数的方式处理异步调用,Promise最大的优势就是可以链式调用解决回调嵌套的问题.但是这样写依然会有大量的回调函数,虽然他们之间没有嵌套,但是还是没有达到 ...
- python初识生成器 迭代器
生成器 带有 yield 的函数在 Python 中被称之为 generator(生成器) def xragns(): #定义函数生成器 print('小伙') yield ('好') #加上yiel ...
随机推荐
- LIS,LCS,LICS 学习笔记
1.最长上升子序列(LIS) 子序列: 1.可以不连续 2.相对位置不变 dp[i][j] 表示前i位置,最大值为j的LIS长度 1. dp[i-1][j] 前i-1位置,最大值为j的LIS长度 (没 ...
- python之简述上下文管理
上下文管理器 原理 代码讲解 原理 上下文管理能保证资源会被正确回收,即保证退出步骤的执行.其用处最多的是,作为确保资源被正确回收的一种方式. 一种重复使用的 try-except-finally 结 ...
- FactoryBean简介
网上看了很多关于FactoryBean和BeanFactory的介绍,总感觉说的不够简单.直白,今天用自己的语言来描述下,如果有不对的地方,还请大家指正. 1. FactoryBean和BeanFac ...
- mysql Access denied for user 'root'@'localhost' (using password: YES)
[现象说明] C/S程序远程訪问正常,本地訪问报下面异常 MySql.Data.MySqlClient.MySqlException (0x80004005): Authentication to h ...
- UML中的序列图(时序图)
序列图将交互关系表示为一个二维图.纵向是时间轴,时间沿竖线向下延伸. 横向轴代表了在协作中各独立对象的类元角色.类元角色用生命线表示.当对象存在时,角色用一条虚线表示,当对象的过程处于激活状态时.生命 ...
- HDU 2181 DFS
Problem Description 一个规则的实心十二面体,它的 20个顶点标出世界著名的20个城市,你从一个城市出发经过每一个城市刚好一次后回到出发的城市. Input 前20行的第i行有3 ...
- EntityFramework 找不到方法:“Void System.Data.Entity.DbModelBuilder.RegisterEntityType
问题原因,EF当前版本没有该方法,将EF版本升级即可. 1.packages.config <package id="EntityFramework" version=&qu ...
- Blur 算法 (Unity 5.0 Shader)
一:简单 Blur 算法 一个像素的颜色值由其邻近的若干像素和自己的颜色值的平均值重新定义,以此达到模糊的效果. 如下图,红色的像素点的值将由它和它周围的24个像素的平均值重新定义.计算的范围一般由一 ...
- redis windows安装与使用
是什么 Redis(Remote Dictionary Server)远程字典服务器 开源免费 C语言编写的 key/value分布式内存数据库,基于内存运行 Redis支持数据的持久化,可以将内存中 ...
- C#+HtmlAgilityPack+Dappe
C#+HtmlAgilityPack+Dappe (转发请注明来源:http://www.cnblogs.com/EminemJK/) 最近因为公司业务需要,又有机会撸winform了,这次的需求是因 ...