php生成器 yield 转
一、yield介绍
文档介绍说道:生成器函数的核心是yield关键字。它最简单的调用形式看起来像一个return申明,不同之处在于普通return会返回值并终止函数的执行,而yield会返回一个值给循环调用此生成器的代码并且只是暂停执行生成器函数。
说了这么多到底是个东西呢,只有自己亲自动手写代码才知道。
二、range函数探索
1、首先看下我们常用的一个产生数组的函数range()
$ran = range(1, 100);
var_dump($ran)//可以看到是一个产生了一个数组
//既然是数组就来遍历看下效果吧
foreach (range(1, 100) as $num) {
echo $num.PHP_EOL;
}
2、现在我们自己来实现这个range()函数
function my_range($start,$limit){
$arr = [];
for($i=$start;$i<=$limit;$i++){
$arr[] =$i;
}
return $arr;
}
了解了这个函数构造后,发现并没有什么神奇的地方,那生成器和这个有什么关系呢,继续往下看。
三、内存占用
1、当我们遍历的范围很小的时候还看不到什么问题,当我们要遍历的数字很大就会超过PHP的内存限制,这下使用生成器就非常有必要了。
2、我们先看上面的range(1, 100)产生的数组占用了多少内存
$start = memory_get_usage();
echo $start.' bytes'.PHP_EOL;
$arr = range(1, 100);
$end = memory_get_usage();
echo $end - $start .' bytes'.PHP_EOL;
结果为:8280 bytes
3、试着增大数字的范围可以看到占用的内存越来越大,下面是我的一组在CLI模式测试的占用内存情况
// 10 728 bytes
// 100 8280 bytes
// 1000 36952 bytes
// 10000 528472 bytes
// 100000 6291568 bytes
// 1000000 35651696 bytes
如果继续增加到10000000,就会报错。根据个人设置PHP内存上限而定。我设置的是512M。
Fatal error: Allowed memory size of 536870912 bytes exhausted (tried to allocate 536870920 bytes) in C:\wamp64\www\phpdemo\yield.php on line 31
4、如果我们就是要操作很大的数字遍历,除了调整内存限制,就无计可施了嘛,这下我们的生成器就要隆重登场了。
四、生成器
1、上面我们自己实现了range()方法,这次我们重新使用生成器来实现这个方法
function my_range($start,$limit){
for($i=$start;$i<=$limit;$i++){
yield $i;
}
}
2、打印出来,看下返回究竟是什么
$arr = my_range(1,100);
var_dump($arr);
- 1
- 2
结果是:
object(Generator)#1 (0) {
}
可见是一个对象,是一个生成器对象,既然是对象那么也就是可以用foreach来遍历
3、遍历生成器
foreach($arr as $num){
echo $num.PHP_EOL;
}
看到可以完整遍历出来,那么与那样实现的不同地方,意义在哪里呢。重点来了。
4、两者内存占用比较
上面已经测试过使用数组的方式,随着范围的增大占用的内存剧增,很快就超过了PHP的内存上限。
那么使用生成器占用了多少内存呢,来看看就知道了。
$start = memory_get_usage();
$arr = my_range(1, 100);
$end = memory_get_usage();
echo $end - $start .' bytes'.PHP_EOL;
可以看到只占用了576bytes,当然每个人测试的可能都会有点不同,环境不同,但是这不是重点。
我们再尝试增加数字范围,可以看到数字范围并没有影响到内存占用,也就是可以轻松的遍历超大数字。
$start = memory_get_usage();
$arr = my_range(1, 100000000);
$end = memory_get_usage();
echo $end - $start .' bytes'.PHP_EOL;
foreach($arr as $num){
echo $num.PHP_EOL;
}
这下我们就可以遍历1到10000000的数字了,不相信内存占用那么低的小伙伴,可以打开任务管理器毫无波澜,即时再上调数字范围。
5、生成器遍历原理
生成器既然这么强大,那么他的遍历原理是什么呢。使用foreach遍历的时候,相当于生成器执行了以下操作。
while($arr->valid()){
echo $arr->current().PHP_EOL;
$arr->next();
}
//$arr->valid() 判断生成器是否关闭
//$arr->current() 返回当前对象
//$arr->next() 继续往下执行生成器
五、结后语
你以为生成器就只有这些用处吗,当然不止。可以使用在很多大量数据的获取场景中,一次性从数据库读取超多的数据,一次性从文本读取超多行文本,都可以这样处理。
不仅用于解决内存问题,还有其他的用武之地,只因本人才疏学浅只能先探索这么多了。
php生成器 yield 转的更多相关文章
- Python入门之迭代器/生成器/yield的表达方式/面向过程编程
本章内容 迭代器 面向过程编程 一.什么是迭代 二.什么是迭代器 三.迭代器演示和举例 四.生成器yield基础 五.生成器yield的表达式形式 六.面向过程编程 ================= ...
- PHP性能优化利器:生成器 yield理解
如果是做Python或者其他语言的小伙伴,对于生成器应该不陌生.但很多PHP开发者或许都不知道生成器这个功能,可能是因为生成器是PHP 5.5.0才引入的功能,也可以是生成器作用不是很明显.但是,生成 ...
- [PY3]——函数——生成器(yield关键字)
函数—生成器篇 1. 认识和区分可迭代or生成器 1.1 可迭代对象 当你建立了一个列表,你可以逐项地读取这个列表,这叫做一个可迭代对象 当你使用一个列表生成式来建立一个列表的时候,就建立了一个可迭代 ...
- 6 生成器 yield 协程
1.生成器 ----> 1 b = [x*2 for x in range(100000000000)] MemoryError: 想生成一个存放很多数据的列表,但是又不想内存占用太多 每次用一 ...
- day4 内置函数 迭代器&生成器 yield总结 三元运算 闭包
内置函数: 内置函数 # abs()返回一个数字的绝对值.如果给出复数,返回值就是该复数的模. b = -100 print(b) print(abs(b)) # all() 所有为真才为真,只要有一 ...
- PHP 生成器 yield理解
如果是做Python或者其他语言的小伙伴,对于生成器应该不陌生.但很多PHP开发者或许都不知道生成器这个功能,可能是因为生成器是PHP 5.5.0才引入的功能,也可以是生成器作用不是很明显.但是,生成 ...
- python3使用迭代生成器yield减少内存占用
技术背景 在python编码中for循环处理任务时,会将所有的待遍历参量加载到内存中.其实这本没有必要,因为这些参量很有可能是一次性使用的,甚至很多场景下这些参量是不需要同时存储在内存中的,这时候就会 ...
- Two---python循环语句/迭代器生成器/yield与return/自定义函数与匿名函数/参数传递
python基础02 条件控制 python条件语句是通过一条或多条语句的执行结果(Ture或者False)来执行的代码块 python中用elif代替了else if,所以if语句的关键字为:if- ...
- Python全栈之路8--迭代器(iter)和生成器(yield)
一.生成器( iter ) 从Python2.2起,生成器提供了一种简洁的方式帮助返回列表元素的函数来完成简单和有效的代码. 它基于yield指令,允许停止函数并立即返回结果.此函数保存其执行上下文, ...
- (转) Python Generators(生成器)——yield关键字
http://blog.csdn.net/scelong/article/details/6969276 生成器是这样一个函数,它记住上一次返回时在函数体中的位置.对生成器函数的第二次(或第 n 次) ...
随机推荐
- Python实现加密的ZIP文件解压(密码已知)
博主在上篇博文介绍了<Python实现加密的RAR文件解压(密码已知)>后,又尝试了ZIP文件的解压方法,下面开始分享. 当ZIP文件的压缩密码已知时,可以通过调用zipfile库进行解压 ...
- Spring Boot 启动第一个页面(Spring二)
在前面的文章里我配置好了spring boot的环境,并选择STS开发工具 具体文章见:https://blog.csdn.net/qq_38175040/article/details/105481 ...
- oracle之序列
序列 15.1 序列是生成唯一整数值的结构,它的典型用途是用于主键值. 结合真题演示伪列nextval, currval用法 CREATE SEQUENCE dept_deptnoINCREMENT ...
- Linux 系统中环境变量/etc/profile、/etc/bashrc、~/.bashrc的区别
/etc/profile./etc/bashrc.~/.bashrc的区别 1> etc目录下存放系统管理和配置文件 (系统配置) etc/profile: profile为所有的用户 ...
- linux定时删除过期文件
需求说明 每日凌晨0点定时删除/temp目录下的所有一个月未被访问的文件. 脚本实现 linux 终端输入crontab -e,添加定时任务脚本命令 [root@localhost ~]# cront ...
- python温度转换代码分析
将用户输入的温度信息保存在TempStr变量中 if分支条件,判断TempStr类型是否在f及F列表之中 如果用户输入的在f及F列表之中,则用户输入的是一个华氏温度值,对华氏温度进行摄氏温度的转换,e ...
- (专题一)04 matlab矩阵表示
矩阵的建立 1. 直接输入法建立矩阵, 建立一个三行三列的矩阵 >>A=[1,2,3;4,5,6;7,8,9] 建立一个三行两列的矩阵 >>B=[1,2;4,5;7,8] 2. ...
- Redis中LIST列表的相关命令
Redis中LIST列表的相关命令 添加 lpush 将一个或多个value插入到key的表头,如果存在多个value,那么各个value按从左到右的顺序依次插入表头 插入表头:意味着新插入的值在最前 ...
- MySQL手注之报错注入
报错注入: 指在页面中没有一个合适的数据返回点的情况下,利用mysql函数的报错来创造一个显位的注入.先来了解一下报错注入常用的函数 XML:指可扩展标记语言被设计用来传输和存储数据. concat: ...
- ACMer不得不会的线段树,究竟是种怎样的数据结构?
大家好,欢迎阅读周三算法数据结构专题,今天我们来聊聊一个新的数据结构,叫做线段树. 线段树这个数据结构很多人可能会有点蒙,觉得没有听说过,但是它非常非常有名,尤其是在竞赛圈,可以说是竞赛圈的必备技能. ...