首頁 後端開發 PHP問題 十分鐘搞清php垃圾回收原理

十分鐘搞清php垃圾回收原理

Sep 04, 2019 am 09:52 AM
php 原理 垃圾回收

十分鐘搞清php垃圾回收原理

php垃圾回收機制,對於PHPer來說是一個不陌生但又不是很熟悉的內容。那麼php是怎麼實現對不需要的記憶體進行回收的呢?

php變數的記憶體儲存結構:

首先還是需要了解下基礎知識,以便於垃圾回收原理內容的理解。大家都知道php是由C寫而成的,所以php變數的內部儲存結構也會和C語言相關,也就是zval的結構體:

struct _zval_struct {
    union {
        long lval;
        double dval;
        struct {
            char *val;
            int len;
        } str;
        HashTable *ht;
        zend_object_value obj;
    } value;                    //变量value值
    zend_uint refcount__gc;   //引用计数内存中使用次数,为0删除该变量
    zend_uchar type;           //变量类型
    zend_uchar is_ref__gc;    //区分是否是引用变量
};
登入後複製

從上面結構體內容可以看出每一個php變數都會由變數類型、value值、引用計數次數和是否是引用變數四部分組成

註:上面zval結構體是php5.3版本之後的結構,php5.3之前因為沒有引進新的垃圾回收機制,即GC,所以命名也沒有_gc;而php7版本之後由於效能問題所以改寫了zval結構,這裡不再表述。

引用計數原則

了解了php變數的內部儲存結構之後,我們再了解下php變數賦值相關的原理與早期垃圾回收機制

變數容器

非array和object變數

每次將常數賦值給一個變數時,都會產生一個變數容器。

範例:

$a = '许铮的技术成长之路';
xdebug_debug_zval('a')
登入後複製

結果:

a: (refcount=1, is_ref=0)='许铮的技术成长之路'
登入後複製

array和object變數

##會產生元素個數1的變數容器

範例:


$b = [
'name' => '许铮的技术成长之路',
'number' => 3
];
xdebug_debug_zval('b')
登入後複製

結果:


b: (refcount=1, is_ref=0)=array ('name' => (refcount=1, is_ref=0)='许铮的技术成长之路', 
'number' => (refcount=1, is_ref=0)=3)
登入後複製

賦值原理(寫時複製技術)

了解了常數賦值之後,接下來我們從記憶體角度思考變數之間的賦值

舉例:

$a = [
'name' => '许铮的技术成长之路',
'number' => 3
]; //创建一个变量容器,变量a指向给变量容器,a的ref_count为1
$b = $a; //变量b也指向变量a指向的变量容器,a和b的ref_count为2
xdebug_debug_zval('a', 'b');
$b['name'] = '许铮的技术成长之路1';//变量b的其中一个元素发生改变,此时会复制出一个新的变量容器,
变量b重新指向新的变量容器,a和b的ref_count变成1
xdebug_debug_zval('a', 'b');
登入後複製

結果:

a: (refcount=2, is_ref=0)=array ('name' => (refcount=1, is_ref=0)='许铮的技术成长之路', 'number' => (refcount=1, is_ref=0)=3)
b: (refcount=2, is_ref=0)=array ('name' => (refcount=1, is_ref=0)='许铮的技术成长之路', 'number' => (refcount=1, is_ref=0)=3)
a: (refcount=1, is_ref=0)=array ('name' => (refcount=1, is_ref=0)='许铮的技术成长之路', 'number' => (refcount=1, is_ref=0)=3)
b: (refcount=1, is_ref=0)=array ('name' => (refcount=1, is_ref=0)='许铮的技术成长之路1', 'number' => (refcount=1, is_ref=0)=3)
登入後複製

所以,當變數a賦值給變數b的時候,並沒有立刻產生一個新的變數容器,而是將變數b指向了變數a指向的變數容器,也就是記憶體"共享";而當變數b其中一個元素改變時,才會真正發生變數容器複製,這就是

寫時複製技術

引用計數清除0

當變數容器的ref_count計數清除時,表示變數容器就會被銷毀,實現了記憶體回收,這也是php5.3版本之前的垃圾回收機制

範例:


$a = "许铮的技术成长之路";
$b = $a;
xdebug_debug_zval('a');
unset($b);
xdebug_debug_zval('a');
登入後複製

結果:


a: (refcount=2, is_ref=0)='许铮的技术成长之路'
a: (refcount=1, is_ref=0)='许铮的技术成长之路'
登入後複製

循環引用引發的記憶體洩漏問題:

#但是php5.3版本之前的垃圾回收機制存在一個漏洞,即當數組或物件內部子元素引用其父元素,而此時如果發生了刪除其父元素的情況,此變數容器並不會被刪除,因為其子元素還在指向該變數容器,但是由於所有作用域內都沒有指向該變數容器的符號,所以無法被清除,因此會發生記憶體洩漏,直到腳本執行結束

舉例:


$a = array( 'one' );
$a[] = &$a;
xdebug_debug_zval( 'a' );
登入後複製

由於該範例不好輸出結果,用圖表表示,如圖:

十分鐘搞清php垃圾回收原理

範例:

unset($a);
xdebug_debug_zval('a');
登入後複製
#如圖:


十分鐘搞清php垃圾回收原理

新的垃圾回收機制:

php5.3版本之後引入根緩衝機制,即php啟動時預設設定指定zval數量的根緩衝區(預設是10000),當php發現有存在循環引用的zval時,就會把其投入到根緩衝區,當根緩衝區達到配置檔案中的指定數量(預設是10000)後,就會進行垃圾回收,以此解決循環引用導致的記憶體洩漏問題

確認為垃圾的準則

1、如果引用計數減少到零,所在變數容器將被清除(free),不屬於垃圾

2、如果一個zval 的引用計數減少後還大於0,那麼它就會進入垃圾週期。其次,在一個垃圾週期中,透過檢查引用計數是否減1,並且檢查哪些變數容器的引用次數是零,來發現哪一部分是垃圾。

總結

垃圾回收機制:

1、以php的引用計數機制為基礎(php5.3以前只有這個機制)
2 、同時使用根緩衝區機制,當php發現有存在循環引用的zval時,就會把其投入到根緩衝區,當根緩衝區達到配置文件中的指定數量後,就會進行垃圾回收,以此解決循環引用導致的記憶體洩漏問題(php5.3開始引入此機制)

更多相關問題請造訪PHP中文網:

php影片教學

以上是十分鐘搞清php垃圾回收原理的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

適用於 Ubuntu 和 Debian 的 PHP 8.4 安裝和升級指南 適用於 Ubuntu 和 Debian 的 PHP 8.4 安裝和升級指南 Dec 24, 2024 pm 04:42 PM

PHP 8.4 帶來了多項新功能、安全性改進和效能改進,同時棄用和刪除了大量功能。 本指南介紹如何在 Ubuntu、Debian 或其衍生版本上安裝 PHP 8.4 或升級到 PHP 8.4

如何設定 Visual Studio Code (VS Code) 進行 PHP 開發 如何設定 Visual Studio Code (VS Code) 進行 PHP 開發 Dec 20, 2024 am 11:31 AM

Visual Studio Code,也稱為 VS Code,是一個免費的原始碼編輯器 - 或整合開發環境 (IDE) - 可用於所有主要作業系統。 VS Code 擁有大量針對多種程式語言的擴展,可以輕鬆編寫

在PHP API中說明JSON Web令牌(JWT)及其用例。 在PHP API中說明JSON Web令牌(JWT)及其用例。 Apr 05, 2025 am 12:04 AM

JWT是一種基於JSON的開放標準,用於在各方之間安全地傳輸信息,主要用於身份驗證和信息交換。 1.JWT由Header、Payload和Signature三部分組成。 2.JWT的工作原理包括生成JWT、驗證JWT和解析Payload三個步驟。 3.在PHP中使用JWT進行身份驗證時,可以生成和驗證JWT,並在高級用法中包含用戶角色和權限信息。 4.常見錯誤包括簽名驗證失敗、令牌過期和Payload過大,調試技巧包括使用調試工具和日誌記錄。 5.性能優化和最佳實踐包括使用合適的簽名算法、合理設置有效期、

php程序在字符串中計數元音 php程序在字符串中計數元音 Feb 07, 2025 pm 12:12 PM

字符串是由字符組成的序列,包括字母、數字和符號。本教程將學習如何使用不同的方法在PHP中計算給定字符串中元音的數量。英語中的元音是a、e、i、o、u,它們可以是大寫或小寫。 什麼是元音? 元音是代表特定語音的字母字符。英語中共有五個元音,包括大寫和小寫: a, e, i, o, u 示例 1 輸入:字符串 = "Tutorialspoint" 輸出:6 解釋 字符串 "Tutorialspoint" 中的元音是 u、o、i、a、o、i。總共有 6 個元

解釋PHP中的晚期靜態綁定(靜態::)。 解釋PHP中的晚期靜態綁定(靜態::)。 Apr 03, 2025 am 12:04 AM

靜態綁定(static::)在PHP中實現晚期靜態綁定(LSB),允許在靜態上下文中引用調用類而非定義類。 1)解析過程在運行時進行,2)在繼承關係中向上查找調用類,3)可能帶來性能開銷。

您如何在PHP中解析和處理HTML/XML? 您如何在PHP中解析和處理HTML/XML? Feb 07, 2025 am 11:57 AM

本教程演示瞭如何使用PHP有效地處理XML文檔。 XML(可擴展的標記語言)是一種用於人類可讀性和機器解析的多功能文本標記語言。它通常用於數據存儲

什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? 什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? Apr 03, 2025 am 12:03 AM

PHP的魔法方法有哪些? PHP的魔法方法包括:1.\_\_construct,用於初始化對象;2.\_\_destruct,用於清理資源;3.\_\_call,處理不存在的方法調用;4.\_\_get,實現動態屬性訪問;5.\_\_set,實現動態屬性設置。這些方法在特定情況下自動調用,提升代碼的靈活性和效率。

PHP和Python:比較兩種流行的編程語言 PHP和Python:比較兩種流行的編程語言 Apr 14, 2025 am 12:13 AM

PHP和Python各有優勢,選擇依據項目需求。 1.PHP適合web開發,尤其快速開發和維護網站。 2.Python適用於數據科學、機器學習和人工智能,語法簡潔,適合初學者。

See all articles