首頁  >  文章  >  後端開發  >  php垃圾回收機制(gc)介紹

php垃圾回收機制(gc)介紹

尚
轉載
2020-03-27 09:17:132858瀏覽

php垃圾回收機制(gc)介紹

php的編譯原理:

ze(zend engine)呼叫詞法分析器把php 程式碼去空格,註解後分割成一個個token,ze呼叫語法分析器再對token處理形成opcode,opcode以op array形式存在,ze最後執行op array輸出結果。

當一個PHP執行緒結束時,目前佔用的所有記憶體空間都會被銷毀。那如果這個線程不結束,怎麼回收記憶體呢?

refcount:引用技術器,可以理解為指向該個容器的指標個數吧。

is_ref:是否被引用(只可能是0或1)

賦值的流程:

<?php

$a = &#39;aa&#39;;
   
xdebug_debug_zval(a);  //(refcount=1, is_ref=0),string &#39;aa&#39; (length=6)

$b = $a; 

//以下的两个其实是一个变量容器

xdebug_debug_zval(a); //(refcount=2, is_ref=0),string &#39;aa&#39; (length=6)
xdebug_debug_zval(b); //(refcount=2, is_ref=0),string &#39;aa&#39; (length=6)

unset($b);  //对变量容器 refcount 减1xdebug_debug_zval(a); //(refcount=1, is_ref=0),string &#39;aa&#39; (length=6)
xdebug_debug_zval(b); //b: no such symbol  b变量被销毁,指向被断掉,如果对应容器的引用技术为零,那么该块儿内存被回收


$b = $a;

$b = &#39;bb&#39;;

xdebug_debug_zval(a); //(refcount=1, is_ref=0),string &#39;aa&#39; (length=6)
xdebug_debug_zval(b); //(refcount=1, is_ref=0),string &#39;aa&#39; (length=6)  重新申请一个变量容器存储b,a的变量容器引用减1

引用的流程:

<?php

$a = &#39;aa&#39;;

xdebug_debug_zval(&#39;a&#39;);  //(refcount=1, is_ref=0),string &#39;aa&#39; (length=2)

$b = & $a;//变量容器的引用技术加1,引用标记置为1xdebug_debug_zval(&#39;a&#39;);  //(refcount=2, is_ref=1),string &#39;aa&#39; (length=2)
xdebug_debug_zval(&#39;b&#39;);  //(refcount=2, is_ref=1),string &#39;aa&#39; (length=2)


$b = &#39;123&#39;; 

//php会发现,该容器变量是引用(is_ref),所以容器变量不用像赋值那样再申请一个

xdebug_debug_zval(&#39;a&#39;);  //(refcount=2, is_ref=1),string &#39;123&#39; (length=2)
xdebug_debug_zval(&#39;b&#39;);  //(refcount=2, is_ref=1),string &#39;123&#39; (length=2)


unset($b);//变量容器应用计数减1,引用为零

xdebug_debug_zval(&#39;a&#39;);  //(refcount=1, is_ref=0),string &#39;123&#39; (length=2)
xdebug_debug_zval(&#39;b&#39;); // b: no such symbol

那麼如果多次引用,unset掉一個,is_ref是否會被置為零,那樣bug不就出現了麼?變數容器還是引用啊。那我們來看看:

<?php


$a = &#39;aa&#39;;

$b = &$a;
$c = &$a;//可以看到引用refCount是3,is_ref永远是1xdebug_debug_zval(&#39;a&#39;); //(refcount=3, is_ref=1),string &#39;aa&#39; (length=2)
xdebug_debug_zval(&#39;b&#39;); //(refcount=3, is_ref=1),string &#39;aa&#39; (length=2)
xdebug_debug_zval(&#39;c&#39;); //(refcount=3, is_ref=1),string &#39;aa&#39; (length=2)


unset($b);//我们期待的bug没有出现,只是refcount减1,is_ref还是1xdebug_debug_zval(&#39;a&#39;); //(refcount=2, is_ref=1),string &#39;aa&#39; (length=2)
xdebug_debug_zval(&#39;b&#39;); // b: no such symbol
xdebug_debug_zval(&#39;c&#39;); //(refcount=2, is_ref=1),string &#39;aa&#39; (length=2)

//那php它怎么知道这个容器还有引用,毕竟is_ref仍然是1,不能计数,那么现在refcount就起作用了,是它告诉php,该变量有几个引用,但问题又来了,如果我干点坏事,在引用的时候,又赋值,它会不会有bug

$e = $a;//我们看到期望的bug还是没出现,这时候再赋值,就不像直接赋值那么简单refcount加1了,而是申请了一个新的变量容器

xdebug_debug_zval(&#39;a&#39;); //(refcount=2, is_ref=1),string &#39;aa&#39; (length=2)
xdebug_debug_zval(&#39;e&#39;); //(refcount=1, is_ref=0),string &#39;aa&#39; (length=2)

unset和賦值null都能回收變數麼?很多人都錯認為,這兩個都能回收變數空間,其實錯了,null只是把變數佔用的空間變小了,從回收來說,該容器依然存在。

<?php

$a = &#39;aa&#39;;

$b = $a;

$b = null;

//又申请了一个变量容器
xdebug_debug_zval(&#39;a&#39;);  //(refcount=1, is_ref=0),string &#39;aa&#39; (length=2)
xdebug_debug_zval(&#39;b&#39;);  //(refcount=1, is_ref=0),null   变量空间并没被回收

unset($b);

//这时候才释放了b变量容器的空间
xdebug_debug_zval(&#39;a&#39;);  //(refcount=1, is_ref=0),string &#39;aa&#39; (length=2)
xdebug_debug_zval(&#39;b&#39;);  //b: no such symbol

推薦:PHP影片教學

#

以上是php垃圾回收機制(gc)介紹的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:oschina.net。如有侵權,請聯絡admin@php.cn刪除