首頁  >  文章  >  後端開發  >  PHP之腳本的記憶體管理與垃圾回收

PHP之腳本的記憶體管理與垃圾回收

不言
不言原創
2018-05-30 09:57:411607瀏覽

這篇文章給大家分享了關於PHP之腳本的內存管理和垃圾回收 ,透過實例來給大家展示了PHP之腳本的內存管理和垃圾回收 ,有需要的朋友可以參考一下。

引用賦值

$a = 'apple';
$b = &$a;

在上述程式碼中,我將一個字串賦值給變數a,然後將a的引用賦值給了變數b。顯然,這個時候的記憶體指向應該是這樣的:

$a -> 'apple' <- $b

a和b指向了同一塊記憶體區域(變數容器),我們透過var_dump($a, $b) 得到string(5) "apple" string(5) "apple" ,這是我們預期的結果。


引用計數 與 unset( )

假如我想將 'apple' 這個字串從記憶體中釋放掉。我是這麼做的:

unset($a);

但是透過再次列印$a $b 兩個變數的訊息,我得到了這樣的結果:Notice: Undefined variable : astring(5) "apple" 。奇怪,$a $b 同時指向一塊記憶體區域,又明明將$a釋放了,為什麼$b還是'apple'

其實是這樣的,unset()這是將一個變數指標銷毀了,並沒有釋放掉那塊記憶體區域(變數容器),所以執行完操作之後,記憶體指向只是變成了這樣:

&#39;apple&#39; <- $b

要牢記重點: unset()並沒有釋放變數所指向的那塊記憶體(變數容器),而只是將變數指標銷毀了。同時,將那塊記憶體的引用計數(ref count) 減1,當引用計數為0時,也就是說當那塊記憶體(變數容器)不被任何變數引用時,便會觸發php的垃圾回收。

用程式碼來驗證一下:

$a = &#39;apple&#39;;
$b = &$a;

$before = memory_get_usage();
unset($a);
$after = memory_get_usage();

var_dump($before - $after);  // 结果为int(0),没有释放
$a = &#39;apple&#39;;
$b = &$a;

$before = memory_get_usage();
unset($a, $b);
$after = memory_get_usage();

var_dump($before - $after);  // 结果为int(24),得到释放

直接回收

那要怎麼做才能真正釋放掉'apple'所佔用的記憶體呢?

利用上述方法,我們可以在unset($a) 之後再unset($b) ,將記憶體區域的所有引用都銷毀,引用計數減為0了,自然就被php回收了。

當然,還有更直接的方法:

$a = null;

直接賦值null 會將$a 所指向的記憶體區域置空,並將引用計數歸零,記憶體便被釋放。


腳本執行結束

php是腳本語言,當腳本執行結束之後,腳本內使用的所有記憶體都會被釋放。那麼,我們手動去釋放記憶體有意義嗎?其實關於這個問題,早有解答,推薦大家看一下鳥哥@laruence 2012年發表的一篇文章:請手動釋放你的資源(Please release resources manually)

#引用賦值

$a = &#39;apple&#39;;
$b = &$a;

在上述程式碼中,我將一個字串賦值給變數a,然後將a的引用賦值給了變數b。顯然,這個時候的記憶體指向應該是這樣的:

$a -> 'apple' <- $b

a和b指向了同一塊記憶體區域(變數容器),我們透過var_dump($a, $b) 得到string(5) "apple" string(5) "apple" ,這是我們預期的結果。


引用計數 與 unset( )

假如我想將 'apple' 這個字串從記憶體中釋放掉。我是這麼做的:

unset($a);

但是透過再次列印$a $b 兩個變數的訊息,我得到了這樣的結果:Notice: Undefined variable : astring(5) "apple" 。奇怪,$a $b 同時指向一塊記憶體區域,又明明將$a釋放了,為什麼$b還是'apple'

其實是這樣的,unset()這是將一個變數指標銷毀了,並沒有釋放掉那塊記憶體區域(變數容器),所以執行完操作之後,記憶體指向只是變成了這樣:

'apple' <- $b

要牢記重點: unset()並沒有釋放變數所指向的那塊記憶體(變數容器),而只是將變數指標銷毀了。同時,將那塊記憶體的引用計數(ref count) 減1,當引用計數為0時,也就是說當那塊記憶體(變數容器)不被任何變數引用時,便會觸發php的垃圾回收。

用程式碼來驗證一下:

$a = 'apple';
$b = &$a;

$before = memory_get_usage();
unset($a);
$after = memory_get_usage();

var_dump($before - $after);  // 结果为int(0),没有释放
$a = 'apple';
$b = &$a;

$before = memory_get_usage();
unset($a, $b);
$after = memory_get_usage();

var_dump($before - $after);  // 结果为int(24),得到释放

直接回收

那要怎麼做才能真正釋放掉'apple'所佔用的記憶體呢?

利用上述方法,我們可以在unset($a) 之後再unset($b) ,將記憶體區域的所有引用都銷毀,引用計數減為0了,自然就被php回收了。

當然,還有更直接的方法:

$a = null;

直接賦值null 會將$a 所指向的記憶體區域置空,並將引用計數歸零,記憶體便被釋放。

以上是PHP之腳本的記憶體管理與垃圾回收的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn