我需要一个PHP CLI脚本,通过GD函数执行各种图像转换。这些图片非常大,所以我需要尽可能地压缩内存。然而,当要求释放内存时,imagedestroy()似乎并没有释放内存。
请考虑以下演示脚本“test.php”:
现在的结果是:
此外,我添加了 。
根据 'free' 命令,内存只有在脚本结束后才会被释放。
请考虑以下演示脚本“test.php”:
#!/usr/bin/php5
<?php
$memory = [0=> memoryCheck()];
$res1 = imagecreatetruecolor(8192, 4096);
memoryReport('Res1 created');
$res2 = imagecreatetruecolor(8192, 4096);
memoryReport('Res2 created');
imagedestroy($res1);
memoryReport('Res1 destroyed');
imagedestroy($res2);
memoryReport('Res2 destroyed');
// memory reporting functions follow:
function memoryCheck()
{
return (int) trim(substr(shell_exec('free -b'), 166, 11));
}
function format($value)
{
$val = \abs($value);
$unit=array('B','kiB','MiB','GiB','TiB','PiB');
return @round($val/pow(1024,($i=floor((($val==0)?0:log($val,1024))))),2).' '.$unit[$i];
}
function memoryReport($msg)
{
global $memory;
$start = $memory[0];
$prev = end($memory);
$now = memoryCheck();
echo sprintf("%s: %s (%s)\n",
$msg,
format($now-$start),
(($diff=$now-$prev) <0) ? '-'. format($diff) : '+'. format($diff)
);
$memory[] = $now;
}
?>
让我们通过以下方式运行它:
free -m;./test.php;free -m
示例结果:
total used free shared buffers cached
Mem: 7890 7072 818 561 218 2497
-/+ buffers/cache: 4355 3534
Swap: 8299 0 8299
Res1 created: 109.76 MiB (+109.76 MiB)
Res2 created: 218.77 MiB (+109.01 MiB)
Res1 destroyed: 218.9 MiB (+140 kiB)
Res2 destroyed: 888 kiB (-218.04 MiB)
total used free shared buffers cached
Mem: 7890 7072 817 561 218 2498
-/+ buffers/cache: 4356 3534
Swap: 8299 0 8299
你可以看到,创建一张图片需要109-110MB。在创建第二张图片后,我们使用的内存翻倍了。但是销毁第一张图片并不会释放内存。只有在两张图片都被销毁后,所有资源的内存才会被释放。
为什么呢?我有遗漏什么吗?我应该怎么做来解决这个问题呢?
更新: 我添加了将$res1设置为null,然后完全取消设置它的代码:
$memory = [0=> memoryCheck()];
$res1 = imagecreatetruecolor(8192, 4096);
memoryReport('Res1 created');
$res2 = imagecreatetruecolor(8192, 4096);
memoryReport('Res2 created');
imagedestroy($res1);
memoryReport('Res1 destroyed');
$res1 = null;
memoryReport('Res1 is null');
unset($res1);
memoryReport('Res1 is unset');
imagedestroy($res2);
memoryReport('Res2 destroyed');
现在的结果是:
Res1 created: 109.48 MiB (+109.48 MiB)
Res2 created: 219.33 MiB (+109.85 MiB)
Res1 destroyed: 219.5 MiB (+168 kiB)
Res1 is null: 220.15 MiB (+668 kiB)
Res1 is unset: 220.38 MiB (+232 kiB)
Res2 destroyed: 2 MiB (-218.36 MiB)
此外,我添加了 。
gc_collect_cycles();
memoryReport('GC collect');
在imagedestroy($res2)之前,一切正常,但接着出现了奇怪的事情:
Res1 created: 109.59 MiB (+109.59 MiB)
Res2 created: 219.08 MiB (+109.5 MiB)
Res1 destroyed: 219.21 MiB (+132 kiB)
Res1 is null: 219.36 MiB (+148 kiB)
Res1 is unset: 219.75 MiB (+408 kiB)
GC collect: 220.57 MiB (+836 kiB)
Res2 destroyed: 220.46 MiB (-108 kiB)
根据 'free' 命令,内存只有在脚本结束后才会被释放。