### 7.7.4 引用計數
在擴展中操作與PHP用戶空間相關的變量時需要考慮是否需要對其引用計數進行加減,比如下面這個例子:
```php
function test($arr){
return $arr;
}
$a = array(1,2);
$b = test($a);
```
如果把函數test()用內部函數實現,這個函數接受了一個PHP用戶空間傳入的數組參數,然后又返回并賦值給了PHP用戶空間的另外一個變量,這個時候就需要增加傳入數組的refcount,因為這個數組由PHP用戶空間分配,函數調用前refcount=1,傳到內部函數時相當于賦值給了函數的參數,因此refcount增加了1變為2,這次增加在函數執行完釋放參數時會減掉,等返回并賦值給$b后此時共有兩個變量指向這個數組,所以內部函數需要增加refcount,增加的引用是給返回值的。test()翻譯成內部函數:
```c
PHP_FUNCTION(test)
{
zval *arr;
if(zend_parse_parameters(ZEND_NUM_ARGS(), "a", &arr) == FAILURE){
RETURN_FALSE;
}
//如果注釋掉下面這句將導致core dumped
Z_TRY_ADDREF_P(arr);
RETURN_ARR(Z_ARR_P(arr));
}
```
那么在哪些情況下需要考慮設置引用計數呢?一個關鍵條件是:操作的是與PHP用戶空間相關的變量,包括對用戶空間變量的修改、賦值,要明確的一點是引用計數是用來解決多個變量指向同一個value問題的,所以在PHP中來回傳遞zval的時候就需要考慮下是不是要修改引用計數,下面總結下PHP中常見的會對引用計數進行操作的情況:
* __(1)變量賦值:__ 變量賦值是最常見的情況,一個用到引用計數的變量類型在初始賦值時其refcount=1,如果后面把此變量又賦值給了其他變量那么就會相應的增加其引用計數
* __(2)數組操作:__ 如果把一個變量插入數組中那么就需要增加這個變量的引用計數,如果要刪除一個數組元素則要相應的減少其引用
* __(3)函數調用:__ 傳參實際可以當做普通的變量賦值,將調用空間的變量賦值給被調函數空間的變量,函數返回時會銷毀函數空間的變量,這時又會減掉傳參的引用,這兩個過程由內核完成,不需要擴展自己處理
* __(4)成員屬性:__ 當把一個變量賦值給對象的成員屬性時需要增加引用計數
PHP中定義了以下宏用于引用計數的操作:
```c
//獲取引用數:pz類型為zval*
#define Z_REFCOUNT_P(pz) zval_refcount_p(pz)
//設置引用數
#define Z_SET_REFCOUNT_P(pz, rc) zval_set_refcount_p(pz, rc)
//增加引用
#define Z_ADDREF_P(pz) zval_addref_p(pz)
//減少引用
#define Z_DELREF_P(pz) zval_delref_p(pz)
#define Z_REFCOUNT(z) Z_REFCOUNT_P(&(z))
#define Z_SET_REFCOUNT(z, rc) Z_SET_REFCOUNT_P(&(z), rc)
#define Z_ADDREF(z) Z_ADDREF_P(&(z))
#define Z_DELREF(z) Z_DELREF_P(&(z))
//只對使用了引用計數的變量類型增加引用,建議使用這個
#define Z_TRY_ADDREF_P(pz) do { \
if (Z_REFCOUNTED_P((pz))) { \
Z_ADDREF_P((pz)); \
} \
} while (0)
#define Z_TRY_DELREF_P(pz) do { \
if (Z_REFCOUNTED_P((pz))) { \
Z_DELREF_P((pz)); \
} \
} while (0)
#define Z_TRY_ADDREF(z) Z_TRY_ADDREF_P(&(z))
#define Z_TRY_DELREF(z) Z_TRY_DELREF_P(&(z))
```
這些宏操作類型都是zval或zval*,如果需要操作具體value的引用計數可以使用以下宏:
```c
//直接獲取zend_value的引用,可以直接通過這個宏修改value的refcount
#define GC_REFCOUNT(p) (p)->gc.refcount
```
另外還有幾個常用的宏:
```c
//判斷zval是否用到引用計數機制
#define Z_REFCOUNTED(zval) ((Z_TYPE_FLAGS(zval) & IS_TYPE_REFCOUNTED) != 0)
#define Z_REFCOUNTED_P(zval_p) Z_REFCOUNTED(*(zval_p))
//根據zval獲取value的zend_refcounted頭部
#define Z_COUNTED(zval) (zval).value.counted
#define Z_COUNTED_P(zval_p) Z_COUNTED(*(zval_p))
```
- 前言
- 第1章 PHP基本架構
- 1.1 PHP簡介
- 1.2 PHP7的改進
- 1.3 FPM
- 1.3.1 概述
- 1.3.2 基本實現
- 1.3.3 FPM的初始化
- 1.3.4 請求處理
- 1.3.5 進程管理
- 1.4 PHP執行的幾個階段
- 第2章 變量
- 2.1 變量的內部實現
- 2.2 數組
- 2.3 靜態變量
- 2.4 全局變量
- 2.5 常量
- 第3章 Zend虛擬機
- 3.1 PHP代碼的編譯
- 3.1.1 詞法解析、語法解析
- 3.1.2 抽象語法樹編譯流程
- 3.2 函數實現
- 3.2.1 內部函數
- 3.2.2 用戶函數的實現
- 3.3 Zend引擎執行流程
- 3.3.1 基本結構
- 3.3.2 執行流程
- 3.3.3 函數的執行流程
- 3.3.4 全局execute_data和opline
- 3.4 面向對象實現
- 3.4.1 類
- 3.4.2 對象
- 3.4.3 繼承
- 3.4.4 動態屬性
- 3.4.5 魔術方法
- 3.4.6 類的自動加載
- 3.5 運行時緩存
- 3.6 Opcache
- 3.6.1 opcode緩存
- 3.6.2 opcode優化
- 3.6.3 JIT
- 第4章 PHP基礎語法實現
- 4.1 類型轉換
- 4.2 選擇結構
- 4.3 循環結構
- 4.4 中斷及跳轉
- 4.5 include/require
- 4.6 異常處理
- 第5章 內存管理
- 5.1 Zend內存池
- 5.2 垃圾回收
- 第6章 線程安全
- 6.1 什么是線程安全
- 6.2 線程安全資源管理器
- 第7章 擴展開發
- 7.1 概述
- 7.2 擴展的實現原理
- 7.3 擴展的構成及編譯
- 7.3.1 擴展的構成
- 7.3.2 編譯工具
- 7.3.3 編寫擴展的基本步驟
- 7.3.4 config.m4
- 7.4 鉤子函數
- 7.5 運行時配置
- 7.5.1 全局變量
- 7.5.2 ini配置
- 7.6 函數
- 7.6.1 內部函數注冊
- 7.6.2 函數參數解析
- 7.6.3 引用傳參
- 7.6.4 函數返回值
- 7.6.5 函數調用
- 7.7 zval的操作
- 7.7.1 新生成各類型zval
- 7.7.2 獲取zval的值及類型
- 7.7.3 類型轉換
- 7.7.4 引用計數
- 7.7.5 字符串操作
- 7.7.6 數組操作
- 7.8 常量
- 7.9 面向對象
- 7.9.1 內部類注冊
- 7.9.2 定義成員屬性
- 7.9.3 定義成員方法
- 7.9.4 定義常量
- 7.9.5 類的實例化
- 7.10 資源類型
- 7.11 經典擴展解析
- 7.8.1 Yaf
- 7.8.2 Redis
- 第8章 命名空間
- 8.1 概述
- 8.2 命名空間的定義
- 8.2.1 定義語法
- 8.2.2 內部實現
- 8.3 命名空間的使用
- 8.3.1 基本用法
- 8.3.2 use導入
- 8.3.3 動態用法
- 附錄
- break/continue按標簽中斷語法實現
- defer推遲函數調用語法的實現
- 一起線上事故引發的對PHP超時控制的思考