- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章PHP中你可能忽略的性能优化利器:生成器由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
前言 。
如果是做python或者其他语言的小伙伴,对于生成器应该不陌生。但很多php开发者或许都不知道生成器这个功能,可能是因为生成器是php 5.5.0才引入的功能,也可以是生成器作用不是很明显。但是,生成器功能的确非常有用.
什么情况之下,会遇到php性能问题?
1:php语法使用不恰当.
2:使用php语言做了它不擅长的事情.
3:使用php语言连接的服务不给力.
4:php自身的短板(php自身做不了的事情).
5:我们也不知道的问题?(去探索、分析找到解决办法,提升开发境界).
优点 。
直接讲概念估计你听完还是一头雾水,所以我们先来说说优点,也许能勾起你的兴趣。那么生成器有哪些优点,如下:
那么,这些神奇的功能究竟是如何做到的?我们先来举个例子.
概念引入 。
首先,放下生成器概念的包袱,来看一个简单的php函数:
1
2
3
4
5
6
7
|
function
createrange(
$number
){
$data
= [];
for
(
$i
=0;
$i
<
$number
;
$i
++){
$data
[] = time();
}
return
$data
;
}
|
这是一个非常常见的php函数,我们在处理一些数组的时候经常会使用。这里的代码也非常简单:
下面没完,我们继续。我们再写一个函数,把这个函数的返回值循环打印出来:
1
2
3
4
5
|
$result
= createrange(10);
// 这里调用上面我们创建的函数
foreach
(
$result
as
$value
){
sleep(1);
//这里停顿1秒,我们后续有用
echo
$value
.
'<br />'
;
}
|
我们在浏览器里面看一下运行结果:
这里非常完美,没有任何问题。(当然sleep(1)效果你们看不出来) 。
思考一个问题 。
我们注意到,在调用函数createrange的时候给$number的传值是10,一个很小的数字。假设,现在传递一个值10000000(1000万).
那么,在函数createrange里面,for循环就需要执行1000万次。且有1000万个值被放到$data里面,而$data数组在是被放在内存内。所以,在调用函数时候会占用大量内存.
这里,生成器就可以大显身手了.
创建生成器 。
我们直接修改代码,你们注意观察:
1
2
3
4
5
|
function
createrange(
$number
){
for
(
$i
=0;
$i
<
$number
;
$i
++){
yield time();
}
}
|
看下这段和刚刚很像的代码,我们删除了数组$data,而且也没有返回任何内容,而是在time()之前使用了一个关键字yield 。
使用生成器 。
我们再运行一下第二段代码:
$result = createrange(10); // 这里调用上面我们创建的函数 foreach($result as $value){ sleep(1); echo $value.'<br />'; } 。
我们奇迹般的发现了,输出的值和第一次没有使用生成器的不一样。这里的值(时间戳)中间间隔了1秒.
这里的间隔一秒其实就是sleep(1)造成的后果。但是为什么第一次没有间隔?那是因为:
到这里,你应该对生成器有点儿头绪.
深入理解生成器 。
代码剖析 。
下面我们来对于刚刚的代码进行剖析.
1
2
3
4
5
6
7
8
9
10
|
function
createrange(
$number
){
for
(
$i
=0;
$i
<
$number
;
$i
++){
yield time();
}
}
$result
= createrange(10);
// 这里调用上面我们创建的函数
foreach
(
$result
as
$value
){
sleep(1);
echo
$value
.
'<br />'
;
}
|
我们来还原一下代码执行过程.
所以,整个代码执行中,始终只有一个记录值参与循环,内存中也只有一条信息.
无论开始传入的$number有多大,由于并不会立即生成所有结果集,所以内存始终是一条循环的值.
概念理解 。
到这里,你应该已经大概理解什么是生成器了。下面我们来说下生成器原理.
首先明确一个概念:生成器yield关键字不是返回值,他的专业术语叫产出值,只是生成一个值 。
那么代码中foreach循环的是什么?其实是php在使用生成器的时候,会返回一个generator类的对象。foreach可以对该对象进行迭代,每一次迭代,php会通过generator实例计算出下一次需要迭代的值。这样foreach就知道下一次需要迭代的值了.
而且,在运行中for循环执行后,会立即停止。等待foreach下次循环时候再次和for索要下次的值的时候,for循环才会再执行一次,然后立即再次停止。直到不满足条件不执行结束.
实际开发应用 。
很多php开发者不了解生成器,其实主要是不了解应用领域。那么,生成器在实际开发中有哪些应用?
读取超大文件 。
php开发很多时候都要读取大文件,比如csv文件、text文件,或者一些日志文件。这些文件如果很大,比如5个g。这时,直接一次性把所有的内容读取到内存中计算不太现实.
这里生成器就可以派上用场啦。简单看个例子:读取text文件 。
我们创建一个text文本文档,并在其中输入几行文字,示范读取.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
<?php
header(
"content-type:text/html;charset=utf-8"
);
function
readtxt()
{
# code...
$handle
=
fopen
(
"./test.txt"
,
'rb'
);
while
(
feof
(
$handle
)===false) {
# code...
yield
fgets
(
$handle
);
}
fclose(
$handle
);
}
foreach
(readtxt()
as
$key
=>
$value
) {
# code...
echo
$value
.
'<br />'
;
}
|
通过上图的输出结果我们可以看出代码完全正常.
但是,背后的代码执行规则却一点儿也不一样。使用生成器读取文件,第一次读取了第一行,第二次读取了第二行,以此类推,每次被加载到内存中的文字只有一行,大大的减小了内存的使用.
这样,即使读取上g的文本也不用担心,完全可以像读取很小文件一样编写代码.
总结 。
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我的支持.
原文链接:https://segmentfault.com/a/1190000012334856 。
最后此篇关于PHP中你可能忽略的性能优化利器:生成器的文章就讲到这里了,如果你想了解更多关于PHP中你可能忽略的性能优化利器:生成器的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我需要将文本放在 中在一个 Div 中,在另一个 Div 中,在另一个 Div 中。所以这是它的样子: #document Change PIN
奇怪的事情发生了。 我有一个基本的 html 代码。 html,头部, body 。(因为我收到了一些反对票,这里是完整的代码) 这是我的CSS: html { backgroun
我正在尝试将 Assets 中的一组图像加载到 UICollectionview 中存在的 ImageView 中,但每当我运行应用程序时它都会显示错误。而且也没有显示图像。 我在ViewDidLoa
我需要根据带参数的 perl 脚本的输出更改一些环境变量。在 tcsh 中,我可以使用别名命令来评估 perl 脚本的输出。 tcsh: alias setsdk 'eval `/localhome/
我使用 Windows 身份验证创建了一个新的 Blazor(服务器端)应用程序,并使用 IIS Express 运行它。它将显示一条消息“Hello Domain\User!”来自右上方的以下 Ra
这是我的方法 void login(Event event);我想知道 Kotlin 中应该如何 最佳答案 在 Kotlin 中通配符运算符是 * 。它指示编译器它是未知的,但一旦知道,就不会有其他类
看下面的代码 for story in book if story.title.length < 140 - var story
我正在尝试用 C 语言学习字符串处理。我写了一个程序,它存储了一些音乐轨道,并帮助用户检查他/她想到的歌曲是否存在于存储的轨道中。这是通过要求用户输入一串字符来完成的。然后程序使用 strstr()
我正在学习 sscanf 并遇到如下格式字符串: sscanf("%[^:]:%[^*=]%*[*=]%n",a,b,&c); 我理解 %[^:] 部分意味着扫描直到遇到 ':' 并将其分配给 a。:
def char_check(x,y): if (str(x) in y or x.find(y) > -1) or (str(y) in x or y.find(x) > -1):
我有一种情况,我想将文本文件中的现有行包含到一个新 block 中。 line 1 line 2 line in block line 3 line 4 应该变成 line 1 line 2 line
我有一个新项目,我正在尝试设置 Django 调试工具栏。首先,我尝试了快速设置,它只涉及将 'debug_toolbar' 添加到我的已安装应用程序列表中。有了这个,当我转到我的根 URL 时,调试
在 Matlab 中,如果我有一个函数 f,例如签名是 f(a,b,c),我可以创建一个只有一个变量 b 的函数,它将使用固定的 a=a1 和 c=c1 调用 f: g = @(b) f(a1, b,
我不明白为什么 ForEach 中的元素之间有多余的垂直间距在 VStack 里面在 ScrollView 里面使用 GeometryReader 时渲染自定义水平分隔线。 Scrol
我想知道,是否有关于何时使用 session 和 cookie 的指南或最佳实践? 什么应该和什么不应该存储在其中?谢谢! 最佳答案 这些文档很好地了解了 session cookie 的安全问题以及
我在 scipy/numpy 中有一个 Nx3 矩阵,我想用它制作一个 3 维条形图,其中 X 轴和 Y 轴由矩阵的第一列和第二列的值、高度确定每个条形的 是矩阵中的第三列,条形的数量由 N 确定。
假设我用两种不同的方式初始化信号量 sem_init(&randomsem,0,1) sem_init(&randomsem,0,0) 现在, sem_wait(&randomsem) 在这两种情况下
我怀疑该值如何存储在“WORD”中,因为 PStr 包含实际输出。? 既然Pstr中存储的是小写到大写的字母,那么在printf中如何将其给出为“WORD”。有人可以吗?解释一下? #include
我有一个 3x3 数组: var my_array = [[0,1,2], [3,4,5], [6,7,8]]; 并想获得它的第一个 2
我意识到您可以使用如下方式轻松检查焦点: var hasFocus = true; $(window).blur(function(){ hasFocus = false; }); $(win
我是一名优秀的程序员,十分优秀!