7

我有一个脚本,用 PHP 编写,使用 AWS Dynamo PHP API。它运行一个长循环,从发电机中提取大量数据,然后对其进行处理。

当我使用“top”查看进程时,我可以看到“php”进程使用的内存使用情况

在我的脚本循环中,我打印 memory_get_usage(true) 的结果

当我运行测试时,这两个值甚至都不相似......

他们应该是吗?如果不是为什么不呢?

在我的测试中,我有一个 1.7gb 内存的服务器,我将 php.ini 的 memory_limit 设置为 64M。我还在脚本开始时调用 gc_enable(),并在每个循环之间调用 gc_collect_cycles(),希望强制进行垃圾回收。

当我使用 'top' 观看我的 php 脚本时,我可以看到 %MEM 不断上升,直到它最终超过 95% 并且 linux 杀死了 php 进程,我通过查看 'dmesg' 知道这一点。当我查看循环每次迭代的打印输出时,memory_get_usage(true) 报告的内存使用量永远不会超过 50mb。

Linux 认为脚本使用了将近 1.7gb,php 认为它只使用了 50mb!

怎么回事?

即使脚本有内存泄漏,我也不明白为什么 memory_get_usage(true) 不考虑内存...

更新

在花了一些时间注释掉我在循环中运行的处理的各个部分后,我发现如果我删除以下代码:

class cMyClass {
    public static function static_cmp_fn(&$a, &$b) {
        if ($a['att'] == $b['att']) { return 0; }
        $ret = ($a['att'] < $b['att']) ? -1 : +1;
        return $ret;
    }
    function DoProcessing(){
        $sort_fn = array("cMyClass", "static_cmp_fn");
        usort($this->m_dictToSort, $sort_fn); 
        unset($sort_fn);
    }

}

php 永远不会吃掉所有的系统内存。在我看来,usort 正在泄漏内存,我不知道为什么。我不明白的是为什么 PHP 报告有关它正在使用多少内存的错误信息......

有任何想法吗?

4

1 回答 1

1

在花了一些时间注释掉我在循环中运行的处理的各个部分后,我发现如果我删除以下代码:

$sort_fn = array("cMyClass", "static_cmp_fn");
usort($this->m_dictToSort, $sort_fn); 

php 永远不会吃掉所有的系统内存。在我看来,usort 正在泄漏内存,我不知道为什么。

显然是的。参见手册:

http://php.net/manual/en/function.usort.php

“这里有几个例子提倡使用 'create_function' 进行排序,由于 usort 的限制,这很容易使用。但要注意这种方法 -创建的函数不会在排序例程结束时被释放,这会创建内存泄漏。因此,可能永远不应该使用这种方法。

array()方法似乎做了类似的事情。您可以声明一个在外部调用您的方法的包装函数,也许?

更新

试图建立一个小测试用例来看看会发生什么。到目前为止,我无法重现泄漏;也许有更多关于什么是结构static_cmp_fn()和结构的数据m_dictToSort。一个简单的比较不会触发任何奇怪的事情。也不会在循环内分配字符串、数组或对象。垃圾收集器将它们杀死并且内存保持在低位。

我会通过调用另一个根本不排序的函数或进行非常基本的排序来进一步限制问题,以查看问题是否在于usort它的可调用函数做一些有趣的事情,正如我所想的那样(它似乎没有,我错了)或者比较函数内部发生了一些有趣的事情。

于 2012-09-28T13:40:55.993 回答