PHP內(nèi)存管理機制
var_dump(memory_get_usage()); //獲取內(nèi)存
$a = "laruence"; //定義一個變量
var_dump(memory_get_usage()); //定義變量之后獲取內(nèi)存
unset($a); //刪除該變量
var_dump(memory_get_usage()); //刪除變量后獲取內(nèi)存
打印結果
int 262144
int 262144
int 262144
從上面可以看出php的內(nèi)存管理機制是:預先給出一塊空間昼牛,用來存儲變量冲茸,當空間不夠時摹芙,再申請一塊新的空間吟温。
1.存儲變量名导狡,存在符號表晾腔。
2.變量值存儲在內(nèi)存空間遗座。
3.在刪除變量的時候豆瘫,會將變量值存儲的空間釋放吞鸭,而變量名所在的符號表不會減小寺董。
var_dump(memory_get_usage()); //獲取內(nèi)存
//定義100個變量
for($i=0;$i<100;$i++)
{ $a = "test".$i;
$a = "hello";
} //獲取定義100個變量之后的內(nèi)存
var_dump(memory_get_usage()); //定義100個變量并刪除
for($i=0;$i<100;$i++)
{ $a = "test".$i; unset($a);
} //獲取刪除之后的內(nèi)存
var_dump(memory_get_usage());</pre>
打印
int 240040
int 257680
int 240856
從上面可以看出,雖然刪除后內(nèi)存變小了刻剥,但還是比沒定義變量之前時大遮咖,這是因為雖然刪除了變量的值,但變量名沒有被刪除造虏。
php垃圾回收機制
PHP變量存儲是存儲在一個zval容器里面的
1.類型 2.值 3.is_ref 代表是否有地址引用 4.refcount 指向該值的變量數(shù)量
1.變量賦值的時候:is_ref為false refcount為1
$a = 1;
xdebug_debug_zval('a');
echo PHP_EOL;
打印結果
(refcount=1, is_ref=0),int 1
2.將變量a的值賦給變量b,變量b不會立刻去在內(nèi)存中存儲值御吞,而是先指向變量a的值,一直到變量a有任何操作的時候
$b = $a;
xdebug_debug_zval('a');
echo PHP_EOL;
打印結果
(refcount=2, is_ref=0), int 1
3.因為程序又操作了變量a漓藕,所以變量b會自己申請一塊內(nèi)存將值放進去陶珠。所以變量a的zavl容器中refcount會減1變?yōu)?,變量c指向a,所以refcount會加1變?yōu)?
$c = &$a;
xdebug_debug_zval('a'); echo PHP_EOL;
xdebug_debug_zval('b'); echo PHP_EOL;
垃圾回收:
1.在5.2版本或之前版本享钞,PHP會根據(jù)refcount值來判斷是不是垃圾
如果refcount值為0揍诽,PHP會當做垃圾釋放掉
這種回收機制有缺陷,對于環(huán)狀引用的變量無法回收
2.在5.3之后版本改進了垃圾回收機制
如果發(fā)現(xiàn)一個zval容器中的refcount在增加,說明不是垃圾
如果發(fā)現(xiàn)一個zval容器中的refcount在減少暑脆,如果減到了0渠啤,直接當做垃圾回收
如果發(fā)現(xiàn)一個zval容器中的refcount在減少,并沒有減到0添吗,PHP會把該值放到緩沖區(qū)沥曹,當做有可能是垃圾的懷疑對象。
當緩沖區(qū)達到了臨界值碟联,PHP會自動調(diào)用一個方法去遍歷每一個值妓美,如果發(fā)現(xiàn)是垃圾就清理