PHP的yield是个什么玩意(一)

其实,我并不是因为迭代或者生成器或者研究PHP手册才认识的yield,要不是协程,我到现在也不知道PHP中还有yield这么个鬼东西。人家这个东西是从PHP 5.5就开始引入了,官方名称叫做生成器。你要说为什么5.5年代的东西,现在才拿出来。我还想问你哟,PHP 5.3就有了的namespace为毛到最近这几年才开始正式投产。

那么,问题来了,这东西到底是有何用?

先来感受一个问题,给你100Kb的内存(是的,你没有看错,就是100Kb),然后让你迭代输出一个从1开始一直到10000的数组,步进为1。

愈先迭代数组,必先创造数组。

所以,脑门一拍,代码一坨如下:

一顿操作猛如虎,运行一下成绩1-5,你们感受一下:

528440bytes,约莫就是528Kb,几乎是100Kb的五倍了,妈的这日子没法过了。

PHP的yield是个什么玩意(一)_第1张图片

毕竟你们也知道,最近内存价格确实贵,国家也在号召低碳节能减排,你多耗费5倍内存,就意味着多排放5倍的二氧化碳,就意味着要为多用的内存多花钱贡献给棒子... ...你想想,那可是棒子。

人都是被逼出来的,于是yield可以来救场了,大概代码如下,注意看操作:

运行一下,你们感受一下:

首先,我们观察一下yield_range这个函数跟普通函数不一样的地方,就是普通函数往往都是使用return来返回结果,而这个中则是yield。其次是普通函数中return只能返回一次,这个yield能返回好多次。

那么,我们来分析一波儿这个神奇的yield_range函数。这个yield关键字到底返回的是什么?我们简单看一下:

yield返回的是一个叫做Generator(中文名就是生成器)的object对象,而这个生成器是实现了Iterator接口(至于Iterator接口,你们去PHP手册上搜索吧)。所以,既然实现了Iterator接口(也正是因为如此,这个东西可以使用foreach进行迭代,明白了吧?),所以可以有如下代码:

valid() ){
  echo $generator->current().PHP_EOL;
  $generator->next();
}

运行结果如下所示:

PHP的yield是个什么玩意(一)_第2张图片

重点来了:这个yield_range函数似乎能够记住它上一次运行到哪儿了,上一次运行的结果是什么,然后紧接着在下一次运行的时候继续从上次终止的地方继续开始。这不是普通的PHP函数可以做得到的!

我们知道,操作系统在调度进程的时候,会触发一个叫做“进程上下文切换”的概念。比如CPU从进程A调度给进程B了,那么当再次从进程B调度给进程A的时候,当初进程A运行到哪儿了、临时的数据结果是什么都是需要被还原的,不然,一切都要从头,那就要出大问题了。而,这个yield关键字,似乎在用户态(非系统内核级)就可以实现这个概念。所以说,用yield搞迭代,怕是真的很没出息的一件事,它能做的太多。

紧接着,我们需要认识一个生成器对象的一个方法,叫做send,简单看下下面这坨代码:

send( $generator->current() * 10 );

运行结果如图所示:

send方法可以修改yield的返回值,但是,你也不能想当然,比如下面这坨代码,你们以为运行结果是什么样呢?

send( $generator->current() * 10 );
}

本来以为运行结果是类似于这样的:

然而,唯物主义告诉我们:

PHP的yield是个什么玩意(一)_第3张图片

结果是打脸的,你们感受一下:

PHP的yield是个什么玩意(一)_第4张图片

原因是什么呢?原因是当你在外部向yield发送send的时候,会自动触发一次next,自己动手试下吧。

你可能感兴趣的:(协程,yield,php)