① php 銷毀變數用unset還是null,什麼時候銷毀
推薦用unset。unset是釋放資源,null是賦空,unset 和 null 的效果是一樣的。
php雖然有自動釋放的機制,但最好是及時釋放。像這樣連續幾天不釋放,會越來越大。
釋放的條件:當一個變數達到目的後,就立即釋放。
純粹的字元變數一般沒有釋放的必要;其他的一些變數,比較大的,或者連庫的,應該立即釋放。
② php把索引數組的第一個元素移除後索引不重置
直接使用php內置函數unset,代碼如下:
//測試數組
$a1=array(1,2,3);
//刪除索引數組第一個值$a1[0]
unset($a1[0]);
//測試結果
echo"<pre>";
print_r($a1);
echo"</pre>";
exit;
結果為:
Array
(
[1] => 2
[2] => 3
)
③ 解析PHP中的unset究竟會不會釋放內存
PHP中的unset究竟會不會釋放內存?以下我們實例說明一下首先讓我們看一個例子
復制代碼 代碼如下: var_mp(memory_get_usage()); $a = "laruence"; var_mp(memory_get_usage()); unset($a); var_mp(memory_get_usage());輸出(在我的個人電腦上 可能會因為系統 PHP版本 載入的擴展不同而不同): int( ) int( ) int(
注意到 = 於是就有了各種的結論 有的人說PHP的unset並不真正釋放內存 有的說 PHP的unset只是在釋放大變數(大量字元串 大數組)的時候才會真正free內存 更有人說 在PHP層面討論內存是沒有意義的
那麼 到底unset會不會釋放內存? 這 個位元組跑哪裡去了? 要回答這個問題 我將從倆個方面入手: 這 個位元組去哪裡了 首先 我們要打破一個思維: PHP不像C語言那樣 只有你顯示的調用內存分配相關API才會有內存的分配 也就是說 在PHP中 有很多我們看不到的內存分配過程 比如對於:
復制代碼 代碼如下: $a = "laruence";隱式的內存分配點就有: 為變數名分配內存 存入符號表 為變數值分配內 所以 不能只看錶象 第二 別懷疑 PHP的unset確實會釋放內存(當然 還要結合引用和計數 這部分的內容請參看我之前的文章深入理解PHP原理之變數分離/引用) 但這個釋放不是C編程意義上的釋放 不是交回給OS 對於PHP來說 它自身提供了一套和C語言對內存分配相似的內存管理API 這些API和C的API意義對應 在PHP內部都是通過這些API來管理內存的
當我們調用emalloc申請內存的時候 PHP並不是簡單的向OS要內存 而是會像OS要一個大塊的內存 然後把其中的一塊分配給申請者 這樣當再有邏輯來申請內存的時候 就不再需要向OS申請內存了 避免了頻繁的系統調用 比如如下的例子:
復制代碼 代碼如下: <?php var_mp(memory_get_usage(TRUE)); //注意獲取的是real_size $a = "laruence"; var_mp(memory_get_usage(TRUE)); unset($a); var_mp(memory_get_usage(TRUE));輸出: int( ) int( ) int(
也就是我們在定義變數$a的時候 PHP並沒有向系統申請新內存
同樣的 在我們調用efree釋放內存的時候 PHP也不會把內存還給OS 而會把這塊內存 歸入自己維護的空閑內存列表 而對於小塊內存來說 更可能的是 把它放到內存緩存列表中去(後記 某些版本的PHP 比如我驗證過的PHP 在調用get_memory_usage()的時候 不會減去內存緩存列表中的可用內存塊大小 導致看起來 unset以後內存不變)
現在讓我來回答這 個位元組跑哪裡去了 就向我剛才說的 很多內存分配的過程不是顯式的 看了下面的代碼你就明白了:
復制代碼 代碼如下: <?php var_mp("I am jb net"); var_mp(memory_get_usage()); $a = "laruence"; var_mp(memory_get_usage()); unset($a); var_mp(memory_get_usage());輸出: string( ) "I am jb net" int( ) //賦值前 int( ) int( ) //是的 內存正常釋放
= 正常了 也就是說這 個位元組是被輸出函數給佔用了(嚴格來說 是被輸出的Header佔用了) 只增不減的數組 Hashtable是PHP的核心結構(了解Hashtable 可以參看我之前的文章深入理解PHP之數組(遍歷順序)) 數組也是用她來表示的 而符號表也是一種關聯數組 對於如下代碼:
復制代碼 代碼如下: var_mp("I am jb net"); var_mp(memory_get_usage()); $array = array_fill( "laruence"); foreach ($array as $key => $value) { ${$value $key} = NULL; } var_mp(memory_get_usage()); foreach ($array as $key=> $value) { unset(${$value $key}); } var_mp(memory_get_usage());我們定義了 個變數 然後又按個Unset了他們 來看看輸出: string( ) "I am jb net" int( ) int( ) int(
Wow 怎麼少了這么多內存? 這是因為對於Hashtable來說 定義它的時候 不可能一次性分配足夠多的內存塊 來保存未知個數的元素 所以PHP會在初始化的時候 只是分配一小部分內存塊給HashTable 當不夠用的時候再RESIZE擴容
而Hashtable 只能擴容 不會減少 對於上面的例子 當我們存入 個變數的時候 符號表不夠用了 做了一次擴容 而當我們依次unset掉這 個變數以後 變數佔用的內存是釋放了( – ) 但是符號表並沒有縮小 所以這些少的內存是被符號表本身佔去了…
lishixin/Article/program/PHP/201311/21152