php5和php7的垃圾回收機制都是利用引用計數,由於PHP是用C來寫的,C裡面有一種東西叫做結構體,我們PHP的變數在C中就是用這種方式儲存的。今天我們就來學習PHP5和PHP7的垃圾回收機制的差異。
php5和php7的垃圾回收機制都是利用引用計數
我們先來看看什麼叫做引用計數:
由於PHP是用C來寫的,C裡面有一種東西叫做結構體,我們PHP的變數在C中就是用這種方式儲存的
每個PHP的變數都存在於一個叫做zval的容器中,一個zval容器,除了包含變數名稱和值,還包括兩個位元組的額外資訊,一個叫做'is_ref',是個布林值,用來表示這個變數是否屬於引用集合,透過這個位元組,我們php才能把普通變數和引用變數區分開來.第二個額外位元組就是'refcount',用來表示指向這個容器的變數的個數
#我們在PHP中定義一個變數
$name='看看';
我們現在可以使用xdebug_debug_zval()函數來獲得函數內變數的相關資訊
xdebug_debug_zval('name'); //输出 name:(refcount=1,is_ref=0)='new String' 现在我们把$name赋值给$money $money = $name;
再來查看一下:
xdebug_debug_zval('name'); //输出 name:(refcount=2,is_ref=0)='new String'
這時候我們會看到refcount的欄位的值增加了1,代表現有兩個變數指向了相同的一個名字叫做name的aval容器
根據引用計數的規則,當這個refcount=0的時候,php就會把這個容器當做垃圾進行回收.
同樣我們執行
unset($name); //一样也会把name容器的引用计数设置为0
以上的結論只是針對標量類型的,下面我們來看看負責類型的
$person=['name'=>'请欢','age'=>19]; var_dump(xdebug_debug_zval('person')); 在PHP5输出: person: (refcount=1, is_ref=0), array (size=2) 'name' => (refcount=1, is_ref=0),string '看看' (length=6) 'age' => (refcount=1, is_ref=0),int 19在PHP7中輸出:
person: (refcount=2, is_ref=0) array (size=2) 'name' => (refcount=1, is_ref=0)string '看看' (length=6) 'age' => (refcount=0, is_ref=0)int 19由此可見,對於複雜的資料型別,PHP5和PHP7的引用計數演算法是不一樣的我們來試試循環引用的情況在上面程式碼的基礎上,新增一行程式碼
$person['hello'] = $person['name']在PHP7中輸出:
person: (refcount=1, is_ref=0) array (size=3) 'name' => (refcount=3, is_ref=0)string '看看' (length=6) 'age' => (refcount=0, is_ref=0)int 19 'hello' => (refcount=3, is_ref=0)string '看看' (length=6)在PHP5中輸出:
person: (refcount=1, is_ref=0), array (size=3) 'name' => (refcount=2, is_ref=0),string '看看' (length=6) 'age' => (refcount=1, is_ref=0),int 19 'hello' => (refcount=2, is_ref=0),string '看看' (length=6)小總結:PHP5和PHP7的垃圾回收機制都屬於引用計數,但是在複雜資料類型的演算法處理上:在PHP7 中zval 有了新的實作方式。最基礎的變化就是 *zval 需要的記憶體不再是單獨從堆上分配,不再自己儲存引用計數。複雜資料類型(例如字串、陣列和物件)的參考計數由其自身來儲存。這種實作方式有以下好處:**簡單資料型別不需要單獨分配內存,也不需要計數;**不會再有兩次計數的情況。在物件中,只有物件本身儲存的計數是有效的;**由於現在計數由數值自身存儲,所以也就可以和非zval 結構的資料共享,例如zval 和hashtable key 之間; *推薦學習:
以上是PHP5和PHP7的垃圾回收機制有什麼不同的詳細內容。更多資訊請關注PHP中文網其他相關文章!