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