<dfn id="w48us"></dfn><ul id="w48us"></ul>
  • <ul id="w48us"></ul>
  • <del id="w48us"></del>
    <ul id="w48us"></ul>
  • PHP對象相互引用的內存溢出實例分析

    時間:2024-07-26 16:32:57 PHP 我要投稿
    • 相關推薦

    關于PHP對象相互引用的內存溢出實例分析

      通常來說使用腳本語言最大的好處之一就是可利用其擁有的自動垃圾回收機制來釋放內存。你不需要在使用完變量后做任何釋放內存的處理,因為這些PHP會幫你完成。

      當然,我們可以按自己的意愿調用 unset() 函數來釋放內存,但通常不需要這么做。

      不過在PHP里,至少有一種情況內存不會得到自動釋放,即便是手動調用 unset()。詳情可考PHP官網關于內存泄露的分析:http://bugs.php.net/bug.php?id=33595。

      問題癥狀如下:

      如果兩個對象之間存在著相互引用的關系,如“父對象-子對象”,對父對象調用 unset()不會釋放在子對象中引用父對象的內存(即便父對象被垃圾回收,也不行)。

      是不是有些糊涂了?我們來看下面的這段代碼:

      bar = new Bar($this); }}class Bar { function __construct($foo = null){ $this->foo = $foo; }}while (true) { $foo = new Foo(); unset($foo); echo number_format(memory_get_usage()) . " ";}?>

      運行這段代碼,你會看到內存使用率越來越高越來越高,直到用光光。

      ...33,551,61633,551,97633,552,33633,552,696PHP Fatal error: Allowed memory size of 33554432 bytes exhausted(tried to allocate 16 bytes) in memleak.php on line 17

      對大部分PHP程序員來講這種情況不算是什么問題。可如果你在一個長期運行的代碼中使用到了一大堆相互引用的對象,尤其是在對象相對較大的情況下,內存會迅速地消耗殆盡。

      Userland解決方案

      雖然有些乏味、不優雅,但之前提到的 bugs.php.net 鏈接中提供了一個解決方案。

      這個方案在釋放對象前使用一個 destructor 方法以達到目的。Destructor 方法可將所有內部的父對象引用全部清除,也就是說可以將這部分本來會溢出的內存釋放掉。

      以下是“修復后”的代碼:

      bar = new Bar($this); } function __destruct(){ unset($this->bar); }}class Bar { function __construct($foo = null){ $this->foo = $foo; }}while (true) { $foo = new Foo(); $foo->__destruct(); unset($foo); echo number_format(memory_get_usage()) . " ";}?>

      注意那個新增的Foo::__destruct()方法,以及在釋放對象前對 $foo->__destruct() 的調用。現在這段代碼解決了內存使用率一直增加的問題,這么一來,代碼就可以很好的工作了。

      PHP內核解決方案

      為什么會有內存溢出的發生?我對PHP內核方面的研究并不精通,但可以確定的是此問題與引用計數有關系。

      在 $bar 中引用 $foo 的引用計數不會因為父對象 $foo 被釋放而遞減,這時PHP認為你仍需要 $foo 對象,也就不會釋放這部分的內存。原理大致如此。

      通俗的來說,大體意思是:一個引用計數沒有遞減,所以一些內存永遠得不到釋放。

      此外在前面提到的 bugs.php.net 鏈接中指出了修改垃圾回收的過程將會犧牲極大的性能,需要讀者對此注意。

      與其改變垃圾回收的過程,為什么不用 unset() 對內部對象做釋放的工作呢?(或者在釋放對象的時候調用 __destruct()?)

      也許PHP內核開發者可以在此或其他地方,對這種垃圾回收處理機制做出修改。

      

    【PHP對象相互引用的內存溢出實例分析】相關文章:

    PHP對象注入的實例分析08-27

    Java內存溢出的類型10-03

    淺析php函數的實例06-08

    PHP中curl的使用實例07-31

    內存故障分析09-01

    關于深入PHP內存相關的功能特性詳解09-02

    PHP面向對象重載重寫的不同10-25

    德國留學申請實例分析09-28

    PHP中的排序函數區別分析08-23

    Javascript 閉包引起IE內存泄露分析07-05

    主站蜘蛛池模板: 亚洲精品国产成人专区| 国产精品天干天干在线综合| 亚洲国产精品婷婷久久| 区亚洲欧美一级久久精品亚洲精品成人网久久久久 | 久99久无码精品视频免费播放| 国产成人精品手机在线观看| 久久九九久精品国产免费直播| 欧美精品国产一区二区| 国产乱码精品一品二品| 伊人久久综合精品无码AV专区| 国产精品无码久久久久| 国产精品高清一区二区三区不卡 | 欧美精品一二区| 亚洲精品国产成人99久久| 久久久一本精品99久久精品88| 久久久无码精品亚洲日韩软件| 影视网欧洲精品| 国产精品三级在线| 久久ww精品w免费人成| 日韩经典精品无码一区| 久久国产精品99久久久久久老狼| 久久精品天天中文字幕人妻| 亚洲国产精品狼友中文久久久 | 亚洲国产精品视频| 国产三级精品三级在专区 | 免费人欧美日韩在线精品| 国产精品免费久久久久久久久| 久久久精品一区二区三区| 大伊香蕉精品视频在线导航| 精品无码一区二区三区爱欲九九 | 精品人体无码一区二区三区| 国产成人精品久久| 99久久精品国产一区二区| 久久青青草原精品影院| 国产精品欧美一区二区三区不卡| 99久久免费国产精品热| 国产精品对白交换视频| 国产精品特级毛片一区二区三区| 精品人妻中文av一区二区三区| 久久亚洲精精品中文字幕| 久久久精品人妻一区二区三区四|