- 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的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
比较代码: const char x = 'a'; std::cout > (0C310B0h) 00C3100B add esp,4 和 const i
您好,我正在使用 Matlab 优化求解器,但程序有问题。我收到此消息 fmincon 已停止,因为目标函数值小于目标函数限制的默认值,并且约束满足在约束容差的默认值范围内。我也收到以下消息。警告:矩
处理Visual Studio optimizations的问题为我节省了大量启动和使用它的时间 当我必须进行 J2EE 开发时,我很难回到 Eclipse。因此,我还想知道人们是否有任何提示或技巧可
情况如下:在我的 Excel 工作表中,有一列包含 1-name 形式的条目。考虑到数字也可以是两位数,我想删除这些数字。这本身不是问题,我让它工作了,只是性能太糟糕了。现在我的程序每个单元格输入大约
这样做有什么区别吗: $(".topHorzNavLink").click(function() { var theHoverContainer = $("#hoverContainer");
这个问题已经有答案了: 已关闭11 年前。 Possible Duplicate: What is the cost of '$(this)'? 我经常在一些开发人员代码中看到$(this)引用同一个
我刚刚结束了一个大型开发项目。我们的时间紧迫,因此很多优化被“推迟”。既然我们已经达到了最后期限,我们将回去尝试优化事情。 我的问题是:优化 jQuery 网站时您要寻找的最重要的东西是什么。或者,我
所以我一直在用 JavaScript 编写游戏(不是网络游戏,而是使用 JavaScript 恰好是脚本语言的游戏引擎)。不幸的是,游戏引擎的 JavaScript 引擎是 SpiderMonkey
这是我在正在构建的页面中使用的 SQL 查询。它目前运行大约 8 秒并返回 12000 条记录,这是正确的,但我想知道您是否可以就如何使其更快提出可能的建议? SELECT DISTINCT Adve
如何优化这个? SELECT e.attr_id, e.sku, a.value FROM product_attr AS e, product_attr_text AS a WHERE e.attr
我正在使用这样的结构来测试是否按下了所需的键: def eventFilter(self, tableView, event): if event.type() == QtCore.QEven
我正在使用 JavaScript 从给定的球员列表中计算出羽毛球 double 比赛的所有组合。每个玩家都与其他人组队。 EG。如果我有以下球员a、b、c、d。它们的组合可以是: a & b V c
我似乎无法弄清楚如何让这个 JS 工作。 scroll function 起作用但不能隐藏。还有没有办法用更少的代码行来做到这一点?我希望 .down-arrow 在 50px 之后 fade out
我的问题是关于用于生产的高级优化级联样式表 (CSS) 文件。 多么最新和最完整(准备在实时元素中使用)的 css 优化器/最小化器,它们不仅提供删除空格和换行符,还提供高级功能,如删除过多的属性、合
我读过这个: 浏览器检索在 中请求的所有资源开始呈现 之前的 HTML 部分.如果您将请求放在 中section 而不是,那么页面呈现和下载资源可以并行发生。您应该从 移动尽可能多的资源请求。
我正在处理一些现有的 C++ 代码,这些代码看起来写得不好,而且调用频率很高。我想知道我是否应该花时间更改它,或者编译器是否已经在优化问题。 我正在使用 Visual Studio 2008。 这是一
我正在尝试使用 OpenGL 渲染 3 个四边形(1 个背景图,2 个 Sprite )。我有以下代码: void GLRenderer::onDrawObjects(long p_dt) {
我确实有以下声明: isEnabled = false; if(foo(arg) && isEnabled) { .... } public boolean foo(arg) { some re
(一)深入浅出理解索引结构 实际上,您可以把索引理解为一种特殊的目录。微软的SQL SERVER提供了两种索引:聚集索引(clustered index,也称聚类索引、簇集索引)和非聚集索引(no
一、写在前面 css的优化方案,之前没有提及,所以接下来进行总结一下。 二、具体优化方案 2.1、加载性能 1、css压缩:将写好的css进行打包,可以减少很多的体积。 2、css单一样式:在需要下边
我是一名优秀的程序员,十分优秀!