亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

PHP垃圾回收機制的案例分析

發布時間:2020-11-09 14:02:34 來源:億速云 閱讀:128 作者:小新 欄目:編程語言

這篇文章將為大家詳細講解有關PHP垃圾回收機制的案例分析,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

 PHP垃圾回收機制

1. PHP可以自動進行內存管理,清除不需要的對象,主要使用了引用計數

2. 在zval結構體中定義了ref_count和is_ref , ref_count是引用計數 ,標識此zval被多少個變量引用 , 為0時會被銷毀。is_ref標識是否使用的 &取地址符強制引用

3. 為了解決循環引用內存泄露問題 , 使用同步周期回收算法。

比如當數組或對象循環的引用自身 , unset掉數組的時候 , 當refcount-1后還大于0的 , 就會被當成疑似垃圾 , 會進行遍歷 ,并且模擬的刪除一次refcount-1如果是0就刪除 ,如果不是0就恢復頑固垃圾的產生過程:

<?php
    $a = "new string";
?>
a: (refcount_gc=1, is_ref_gc=0)='new string'

當把$a賦值給另外一個變量的時候,$a對應的zval的refcount_gc會加1

<?php
    $a = "new string";
    $b = $a;
?>

此時$a和$b變量對應的內部存儲信息為,$a和$b同時指向一個字符串"new string" ,它的refcount變成2a,b: (refcount_gc=2,is_ref=0)='new string'

當用unset刪除$b變量時,"new string" 的refcount_gc會減1變成1

<?php
    $a = "new string"; //a: (refcount_gc=1, is_ref_gc=0)='new string'
    $b = $a;           //a,b: (refcount_gc=2, is_ref=0)='new string'
    unset($b);         //a: (refcount_gc=1, is_ref=0)='new string'
?>

對于普通的變量來說,這一切很正常,但是在復合類型變量(數組和對象)中,會發生比較有意思的事情:

<?php
    $a = array('meaning' => 'life', 'number' => 42);
?>

$a內部存儲信息為:

a: (refcount=1, is_ref=0)=array (
'meaning' => (refcount=1, is_ref=0)='life',
'number' => (refcount=1, is_ref=0)=42
)

數組變量本身($a)在引擎內部實際上是一個哈希表,這張表中有兩個zval項 meaning和number,所以實際上那一行代碼中一共生成了3個zval,這3個zval都遵循變量的引用和計數原則,用圖來表示:

PHP垃圾回收機制的案例分析

下面在$a中添加一個元素,并將現有的一個元素的值賦給新的元素:

<?php
    $a = array('meaning' => 'life', 'number' => 42);
    $a['name'] = $a['meaning'];
 ?>

那么$a的內部存儲為 , "life" 的ref_count變成2 , 42的ref_count是1:

a: (refcount=1, is_ref=0)=array (
'meaning' => (refcount=2, is_ref=0)='life',
'number' => (refcount=1, is_ref=0)=42,
'name' => (refcount=2, is_ref=0)='life'
)

如果將數組的引用賦值給數組中的一個元素,有意思的事情就會發生:

<?php
    $a = array('one');
    $a[] = &$a;
?>

這樣$a數組就有兩個元素,一個索引為0,值為字符one,另外一個索引為1,為$a自身的引用,內部存儲如下:

PHP垃圾回收機制的案例分析

a: (refcount=2, is_ref=1)=array (
0 => (refcount=1, is_ref=0)='one',
1 => (refcount=2, is_ref=1)=…
)

array這個zval的ref_count是2 , 是一個環形引用。這時對$a進行unset,那么$a會從符號表中刪除,同時$a指向的zval的refcount_gc減少1.

那么問題就產生了,$a已經不在符號表中,用戶無法再訪問此變量,但是$a之前指向的zval的refcount_gc變為1而不是0,因此不能被回收,從而產生內存泄露,新的GC要做的工作就是清理此類垃圾。

為了解決循環引用內存泄露問題 , 使用同步周期回收算法 , 這種ref_count減1后還大于0的會被作為疑似垃圾。

比如當數組或對象循環的引用自身 , unset掉數組的時候 , 當refcount-1后還大于0的 , 會進行遍歷 ,并且模擬的刪除一次refcount-1如果是0就刪除 ,如果不是0就恢復。

關于PHP垃圾回收機制的案例分析就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

php
AI

泾阳县| 西平县| 隆德县| 土默特右旗| 宁波市| 丽江市| 锦屏县| 仁寿县| 正安县| 德州市| 扶余县| 新邵县| 泰和县| 东阳市| 尼勒克县| 湟中县| 辉县市| 乾安县| 易门县| 景德镇市| 哈巴河县| 靖安县| 靖远县| 友谊县| 长武县| 永寿县| 山阳县| 刚察县| 旅游| 临汾市| 汤阴县| 兴海县| 富阳市| 雅安市| 剑阁县| 温州市| 沛县| 堆龙德庆县| 白山市| 汝州市| 九寨沟县|