33
44本节提到的zval泛指zval及各种zend_value。
55
6- ### 7.7.1 生成不同类型的zval
6+ ### 7.7.1 新生成各类型zval
77PHP7将变量的引用计数转移到了具体的value上,所以zval更多的是作为统一的传输格式,很多情况下只是临时性使用,比如函数调用时的传参,最终需要的数据是zval携带的zend_value,函数从zval取得zend_value后就不再关心zval了,这种就可以直接在栈上分配zval。分配完zval后需要将其设置为我们需要的类型以及设置其zend_value,PHP中定义的` ZVAL_XXX() ` 系列宏就是用来干这个的,这些宏第一个参数z均为要设置的zval的指针,后面为要设置的zend_value。
88
99* __ ZVAL_UNDEF(z):__ 表示zval被销毁
@@ -27,7 +27,85 @@ PHP7将变量的引用计数转移到了具体的value上,所以zval更多的
2727* __ ZVAL_NEW_REF(z, r):__ 新创建一个引用,r为引用的值,类型为zval*
2828* ...
2929
30- ### 7.7.2 获取zval的值
30+ ### 7.7.2 获取zval的值及类型
31+ zval的类型通过` Z_TYPE(zval) ` 、` Z_TYPE_P(zval*) ` 两个宏获取,这个值取的就是` zval.u1.v.type ` ,但是设置时不要只修改这个type,而是要设置typeinfo,因为zval还有其它的标识需要设置,比如是否使用引用计数、是否可被垃圾回收、是否可被复制等等。
3132
33+ 内核提供了` Z_XXX(zval) ` 、` Z_XXX_P(zval*) ` 系列的宏用于获取不同类型zval的value。
34+
35+ * __ Z_LVAL(zval)、Z_LVAL_P(zval_p):__ 返回zend_long
36+ * __ Z_DVAL(zval)、Z_DVAL_P(zval_p):__ 返回double
37+ * __ Z_STR(zval)、Z_STR_P(zval_p):__ 返回zend_string*
38+ * __ Z_STRVAL(zval)、Z_STRVAL_P(zval_p):__ 返回char* ,即:zend_string->val
39+ * __ Z_STRLEN(zval)、Z_STRLEN_P(zval_p):__ 获取字符串长度
40+ * __ Z_STRHASH(zval)、Z_STRHASH_P(zval_p):__ 获取字符串的哈希值
41+ * __ Z_ARR(zval)、Z_ARR_P(zval_p)、Z_ARRVAL(zval)、Z_ARRVAL_P(zval_p):__ 返回zend_array*
42+ * __ Z_OBJ(zval)、Z_OBJ_P(zval_p):__ 返回zend_object*
43+ * __ Z_OBJ_HT(zval)、Z_OBJ_HT_P(zval_p):__ 返回对象的zend_object_handlers,即zend_object->handlers
44+ * __ Z_OBJ_HANDLER(zval, hf)、Z_OBJ_HANDLER_P(zv_p, hf):__ 获取对象各操作的handler指针,hf为write_property、read_property等,注意:这个宏取到的为只读,不要试图修改这个值(如:Z_OBJ_HANDLER(obj, write_property) = xxx;),因为对象的handlers成员前加了const修饰符
45+ * __ Z_OBJCE(zval)、Z_OBJCE_P(zval_p):__ 返回对象的zend_class_entry*
46+ * __ Z_OBJPROP(zval)、Z_OBJPROP_P(zval_p):__ 获取对象的成员数组
47+ * __ Z_RES(zval)、Z_RES_P(zval_p):__ 返回zend_resource*
48+ * __ Z_RES_HANDLE(zval)、Z_RES_HANDLE_P(zval_p):__ 返回资源handle
49+ * __ Z_RES_TYPE(zval)、Z_RES_TYPE_P(zval_p):__ 返回资源type
50+ * __ Z_RES_VAL(zval)、Z_RES_VAL_P(zval_p):__ 返回资源ptr
51+ * __ Z_REF(zval)、Z_REF_P(zval_p):__ 返回zend_reference*
52+ * __ Z_REFVAL(zval)、Z_REFVAL_P(zval_p):__ 返回引用的zval*
53+
54+ 除了这些与PHP变量类型相关的宏之外,还有一些内核自己使用类型的宏:
55+ ``` c
56+ // 获取indirect的zval,指向另一个zval
57+ #define Z_INDIRECT (zval ) (zval).value.zv
58+ #define Z_INDIRECT_P (zval_p ) Z_INDIRECT(*(zval_p))
59+
60+ #define Z_CE (zval ) (zval).value.ce
61+ #define Z_CE_P (zval_p ) Z_CE(*(zval_p))
62+
63+ #define Z_FUNC (zval ) (zval).value.func
64+ #define Z_FUNC_P (zval_p ) Z_FUNC(*(zval_p))
65+
66+ #define Z_PTR (zval ) (zval).value.ptr
67+ #define Z_PTR_P (zval_p ) Z_PTR(*(zval_p))
68+ ```
69+ zend_string常用的宏:
70+ ``` c
71+ // zstr类型为zend_string*
72+ #define ZSTR_VAL (zstr ) (zstr)->val
73+ #define ZSTR_LEN(zstr) (zstr)->len
74+ #define ZSTR_H(zstr) (zstr)->h
75+ #define ZSTR_HASH(zstr) zend_string_hash_val(zstr)
76+ ```
3277### 7.7.3 引用计数
78+ ```c
79+ //获取引用数:pz类型为zval*
80+ #define Z_REFCOUNT_P(pz) zval_refcount_p(pz)
81+ //设置引用数
82+ #define Z_SET_REFCOUNT_P(pz, rc) zval_set_refcount_p(pz, rc)
83+ //增加引用
84+ #define Z_ADDREF_P(pz) zval_addref_p(pz)
85+ //减少引用
86+ #define Z_DELREF_P(pz) zval_delref_p(pz)
87+
88+ #define Z_REFCOUNT(z) Z_REFCOUNT_P(&(z))
89+ #define Z_SET_REFCOUNT(z, rc) Z_SET_REFCOUNT_P(&(z), rc)
90+ #define Z_ADDREF(z) Z_ADDREF_P(&(z))
91+ #define Z_DELREF(z) Z_DELREF_P(&(z))
92+
93+ //只对使用了引用计数的变量类型增加引用,建议使用这个
94+ #define Z_TRY_ADDREF_P(pz) do { \
95+ if (Z_REFCOUNTED_P((pz))) { \
96+ Z_ADDREF_P((pz)); \
97+ } \
98+ } while (0)
99+
100+ #define Z_TRY_DELREF_P(pz) do { \
101+ if (Z_REFCOUNTED_P((pz))) { \
102+ Z_DELREF_P((pz)); \
103+ } \
104+ } while (0)
105+
106+ #define Z_TRY_ADDREF(z) Z_TRY_ADDREF_P(&(z))
107+ #define Z_TRY_DELREF(z) Z_TRY_DELREF_P(&(z))
108+ ```
109+
110+ ### 7.7.4 数组操作
33111
0 commit comments