为什么GD在使用imagedestroy()释放内存时无法完全释放?

3
我需要一个PHP CLI脚本,通过GD函数执行各种图像转换。这些图片非常大,所以我需要尽可能地压缩内存。然而,当要求释放内存时,imagedestroy()似乎并没有释放内存。
请考虑以下演示脚本“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' 命令,内存只有在脚本结束后才会被释放。

4
imagedestroy()只会释放PHP内存中可用的空间,不一定会将内存释放回系统。PHP从系统中分配内存块。如果$res1使用的块恰好包含来自$res2(或任何其他变量)的内容,则该块不会被释放。除了尝试更改这些资源创建的顺序外,你没有太多可以做的事情。 - cleong
1
@cleong,你应该把这个评论发表为一个答案。 - AllInOne
确实,请这样做,以便我可以奖励您的回答 :) - Forseti
@cleong:请将您的评论发布为答案,这样我就可以奖励您的答案 :) - Forseti
1个回答

0

我在想这是否是垃圾回收的问题... 在imagedestroy()之后尝试运行gc_collect_cycles()。虽然可能性不大,但你永远不知道...


添加gc_collect_cycles()实际上使情况变得更糟。我现在更加困惑了。 - Forseti
哎呀...抱歉,我已经没有更多的想法了。祝你好运! - Matt Parlane

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接