【PHP內存泄漏案例】PHP對象遞歸引用造成內存泄漏


案例一

作者:老王

如果PHP對象存在遞歸引用,就會出現內存泄漏。這個Bug在PHP里已經存在很久很久了,先讓我們來重現這個Bug,代碼如下:

<?php
class Foo {
    function __construct() {
        $this->bar = new Bar($this);
    }
}

class Bar {
    function __construct($foo) {
        $this->foo = $foo;
    }
}

for ($i = 0; $i < 100; $i++) {
    $obj = new Foo();

    unset($obj);

    echo memory_get_usage(), "
";
}
?>

運行以上代碼,你會發現,內存使用量本應該不變才對,可實際上卻是不斷增加,unset沒有完全生效。

現在的開發很多都是基於框架進行的,應用里存在復雜的對象關系,那么就很可能會遇到這樣的問題,下面看看有什么權宜之計:

<?php
class Foo {
    function __construct() {
        $this->bar = new Bar($this);
    }

    function __destruct() {
        unset($this->bar);
    }
}

class Bar {
    function __construct($foo) {
        $this->foo = $foo;
    }
}

for ($i = 0; $i < 100; $i++) {
    $obj = new Foo();

    $obj->__destruct();

    unset($obj);

    echo memory_get_usage(), "
";
}
?>

辦法有些丑陋,不過總算是對付過去了。幸運的是這個Bug在PHP5.3的CVS代碼中已經被修復了。

參考鏈接:http://bugs.php.net/bug.php?id=33595

 


 

案例二

作者:貴貴的博客

出現了一個PHP腳本占用系統內存50%

開始認為是哪個變量PHP沒有自動釋放,查看代碼,里面有個類的函數是static的,加了設置為Null和unset后無效果。

memory_get_usage方法一測試,才發現在每次數據庫讀取都會增加內存。

看了db類找到原因了,原來db對象是放在全局里的,在數據庫讀取時會在他里面記錄下執行的SQL。全局的程序沒有執行完畢是不會釋放的,這個腳本又是常駐內存執行的,解決方法也簡單取消記錄這個SQL就可以了。

經驗:對於全局變量在循環調用里的內存使用要注意

<?php
class db
{
    public $sql;

    public function query($sql)
    {
        $this->sql[] = $sql;
    }
}

$db = new db();
while(1) {
    echo "m1:".memory_get_usage()."
";
    $db->query("select * from table");
    echo "m2:".memory_get_usage()."
";
    usleep(10000);
}

顯然這不是內存泄漏的BUG,只是代碼編碼的問題。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM