我在 StackOverflow 上找到了以下解决方案,可以从对象数组中获取特定对象属性的数组:PHP - Extracting a property from an array of objects
建议的解决方案是使用 array_map
并在其中使用 create_function
创建一个函数,如下所示:
$catIds = array_map(create_function('$o', 'return $o->id;'), $objects);
会发生什么?:array_map
遍历每个数组元素,在本例中是一个 stdClass
对象。首先它创建一个这样的函数:
function($o) {
return $o->id;
}
其次,它为当前迭代中的对象调用此函数。它有效,它的工作原理与类似的解决方案几乎相同:
$catIds = array_map(function($o) { return $o->id; }, $objects);
但此解决方案仅在 PHP 版本 >= 5.3 中运行,因为它使用了 Closure
概念 => http://php.net/manual/de/class.closure.php
现在真正的问题是:
create_function
的第一个解决方案增加了内存,因为创建的函数将被写入内存,不会被重用或销毁。在带有 Closure
的第二个解决方案中,它会。
所以这些解决方案给出了相同的结果,但在内存方面具有不同的行为。
下面的例子:
// following array is given
$objects = array (
[0] => stdClass (
[id] => 1
),
[1] => stdClass (
[id] => 2
),
[2] => stdClass (
[id] => 3
)
)
不好
while (true)
{
$objects = array_map(create_function('$o', 'return $o->id;'), $objects);
// result: array(1, 2, 3);
echo memory_get_usage() ."\n";
sleep(1);
}
4235616
4236600
4237560
4238520
...
好
while (true)
{
$objects = array_map(function($o) { return $o->id; }, $objects);
// result: array(1, 2, 3);
echo memory_get_usage() ."\n";
sleep(1);
}
4235136
4235168
4235168
4235168
...
我花了很多时间来找出这个问题,现在我想知道,这是垃圾收集器的错误还是我弄错了? 为什么将已创建和调用的函数留在内存中是有意义的,因为它永远不会被重用?
这是一个运行示例:http://ideone.com/9a1D5g
更新:当我递归搜索我的代码及其依赖项时,例如PEAR 和 Zend 然后我经常发现这种BAD方式。
更新:当两个函数嵌套时,我们从内向外进行以评估此表达式。换句话说,它首先启动 create_function
(一次),返回的函数名称是 array_map
的单次调用的参数。但是因为 GC 忘记将它从内存中删除(没有指针留在内存中的函数)并且 PHP 无法重用已经位于内存中的函数让我认为存在错误而不仅仅是“性能差”的事情.这行特定的代码是 PHPDoc 中的示例,并在许多大型框架中重复使用,例如Zend 和 PEAR 等等。再多一行就可以解决这个“错误”,检查一下。但我不是在寻找解决方案:我在寻找真相。这是一个错误还是只是我的方法。后者我还不能决定。
最佳答案
在 create_function()
的情况下,使用 eval()
创建 lambda 样式的函数,并返回包含其名称的字符串。然后将该名称作为参数传递给 array_map()
函数。
这不同于闭包风格的匿名函数,后者根本不使用包含名称的字符串。 函数($o) { 返回$o->id;
是函数,或者更确切地说是 Closure 类的一个实例。
eval()
函数在 create_function()
中,执行一段 PHP 代码来创建所需的函数。有点像这样:
function create_function($arguments,$code)
{
$name = <_lambda_>; // just a unique string
eval('function '.$name.'($arguments){$code}');
return $name;
}
请注意,这是一种简化。
因此,一旦函数被创建,它将一直持续到脚本结束,就像脚本中的普通函数一样。在上面的 BAD 示例中,在循环的每次迭代中都会像这样创建一个新函数,占用越来越多的内存。
但是,您可以故意破坏 lambda 风格的函数。这很简单,只需将循环更改为:
while (true)
{
$func = create_function('$o', 'return $o->id;');
$objects = array_map($func, $objects);
echo memory_get_usage() ."\n";
sleep(1);
}
包含对函数的引用(= 名称)的字符串在这里是明确的和可访问的。现在,每次调用 create_function()
时,旧函数都会被新函数覆盖。
所以,不,没有“内存泄漏”,它本来就是这样工作的。
当然下面的代码效率更高:
$func = create_function('$o', 'return $o->id;');
while (true)
{
$objects = array_map($func, $objects);
echo memory_get_usage() ."\n";
sleep(1);
}
并且只应在您的 PHP 版本不支持闭包式匿名函数时使用。
关于php - 内存泄漏?!在 'create_function' 中使用 'array_map' 时,垃圾收集器是否正常运行?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25808390/