资讯专栏INFORMATION COLUMN

PHP下的异步尝试一:初识生成器

tomorrowwu / 2935人阅读

摘要:下的异步尝试系列下的异步尝试一初识生成器下的异步尝试二初识协程下的异步尝试三协程的版自动执行器下的异步尝试四版的下的异步尝试五版的的继续完善生成器类获取迭代器当前值获取迭代器当前值返回当前产生的键生成器从上一次处继续执行重置迭代器向生成器中

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 开始
";
    yield new Exception("内部定义异常");
    echo "Gen 结束
";
}

$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 开始
";
    throw (yield new Exception("内部定义异常"));
    echo "Gen 结束
";
}

$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 开始
";
    try {
        yield new Exception("内部定义异常");
    } catch (Exception $e) {
        echo "内部捕获异常:" . "Exception: {$e->getMessage()}
";
    }

    echo "Gen 结束
";
}

$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://www.ucloud.cn/yun/30895.html

相关文章

  • PHP下的异步尝试二:初识协程

    摘要:如果仅依靠程序自动交出控制的话,那么一些恶意程序将会很容易占用全部时间而不与其他任务共享。多个操作可以在重叠的时间段内进行。 PHP下的异步尝试系列 如果你还不太了解PHP下的生成器,你可以根据下面目录翻阅 PHP下的异步尝试一:初识生成器 PHP下的异步尝试二:初识协程 PHP下的异步尝试三:协程的PHP版thunkify自动执行器 PHP下的异步尝试四:PHP版的Promise ...

    MudOnTire 评论0 收藏0
  • PHP下的异步尝试四:PHP版的Promise

    摘要:结果打印我结论或问题这里我们基础实现了一个可以用于生产环境的后续我们会接续完善这个的特有方法,比如等后续再介绍用实现的自动执行器等附录参考中文对象入门阮一峰 PHP下的异步尝试系列 如果你还不太了解PHP下的生成器和协程,你可以根据下面目录翻阅 PHP下的异步尝试一:初识生成器 PHP下的异步尝试二:初识协程 PHP下的异步尝试三:协程的PHP版thunkify自动执行器 PHP下的...

    lentoo 评论0 收藏0
  • PHP下的异步尝试三:协程的PHP版thunkify自动执行器

    摘要:函数并不是生成器协程函数自动执行的唯一方案。因为自动执行的关键是,必须有一种机制,自动控制生成器协程函数的流程,接收和交还程序的执行权。回调函数可以做到这一点,对象也可以做到这一点。本系列的下一篇,将介绍基于的实现的自动执行器。 PHP下的异步尝试系列 如果你还不太了解PHP下的生成器和协程,你可以根据下面目录翻阅 PHP下的异步尝试一:初识生成器 PHP下的异步尝试二:初识协程 P...

    wudengzan 评论0 收藏0
  • 初识wkhtmltopdf

    摘要:背景最近接到一个客户需求,要求将学生的考试结果分析表格和图表导出到。本着最好不要额外安装软件的原则,搜索过后分别尝试了等等。但是实现效果都与预期差距较大。它们是开源命令行工具,使用渲染引擎将呈现为和各种图像格式。它们不需要显示或显示服务。 背景 最近接到一个客户需求,要求将学生的考试结果分析表格和图表导出到PDF。表格使用的是普通的table,图表引用了https://www.char...

    Towers 评论0 收藏0
  • 初识wkhtmltopdf

    摘要:背景最近接到一个客户需求,要求将学生的考试结果分析表格和图表导出到。本着最好不要额外安装软件的原则,搜索过后分别尝试了等等。但是实现效果都与预期差距较大。它们是开源命令行工具,使用渲染引擎将呈现为和各种图像格式。它们不需要显示或显示服务。 背景 最近接到一个客户需求,要求将学生的考试结果分析表格和图表导出到PDF。表格使用的是普通的table,图表引用了https://www.char...

    wushuiyong 评论0 收藏0

发表评论

0条评论

tomorrowwu

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<