为什么数组引用自身造成内存泄露?


有下面一段代码:

   
  echo memory_get_usage(),"\n";
  
$arr = array(str_pad('a',3072,'a'));
$arr[1] = &$arr;
echo memory_get_usage(),"\n";
unset($arr);
echo memory_get_usage(),"\n";

理论上最后一次输出的值应该和第一次相同或者接近,但是事实上后两次内存值相近,理论上unset($arr)之后,内存应该收回,但是此时并没有收回,造成了内存泄露,为什么?

内存管理 php

桃子味气球 12 years, 4 months ago

PHP的unset确实会释放内存(当然, 还要结合引用和计数, 但这个释放不是C编程意义上的释放, 不是交回给OS.
对于PHP来说, 它自身提供了一套和C语言对内存分配相似的内存管理API:

   
  emalloc(size_t size);
  
efree(void *ptr);
ecalloc(size_t nmemb, size_t size);
erealloc(void *ptr, size_t size);
estrdup(const char *s);
estrndup(const char *s, unsigned int length);

这些API和C的API意义对应, 在PHP内部都是通过这些API来管理内存的.

当我们调用emalloc申请内存的时候, PHP并不是简单的向OS要内存, 而是会像OS要一个大块的内存, 然后把其中的一块分配给申请者, 这样当再有逻辑来申请内存的时候, 就不再需要向OS申请内存了, 避免了频繁的系统调用.

   
  echo memory_get_usage(true),"\n";
  
$arr = array(str_pad('a',3072,'a'));
$arr[1] = &$arr;
echo memory_get_usage(true),"\n";
unset($arr);
echo memory_get_usage(true),"\n";

输出的值都是一样的。
也就是我们在定义变量$arr的时候, PHP并没有向系统申请新内存.
同样的, 在我们调用efree释放内存的时候, PHP也不会把内存还给OS, 而会把这块内存, 归入自己维护的空闲内存列表. 而对于小块内存来说, 更可能的是, 把它放到内存缓存列表中去。
参考http://www.laruence.com/2011/03/04/1894.html

jzjgxsh answered 12 years, 4 months ago

Your Answer