typedef union _zvalue_value {
long lval;
double dval;
struct {
char *val;
int len;
} str;
HashTable *ht;
zend_object obj;
} zvalue_value;
//pzval和zval的定义:
struct _zval_struct {
zvalue_value value;
zend_uchar type;
zend_uchar is_ref;
zend_ushort refcount;
};
下面介绍下zval相关的宏
MAKE_STD_ZVAL()
这个可能是使用的最多的宏了,因为它的工作就是初始化一个zval,代码实例:
zval *var;
MAKE_STD_ZVAL(var);
通过下面这源码,应该很清楚MAKE_STD_ZVAL都干了些什么了吧。分配内存,置refcount为1,置is_ref为0。
// MAKE_STD_ZVAL定义在zend.h里
#define MAKE_STD_ZVAL(zv) \
ALLOC_ZVAL(zv); \
INIT_PZVAL(zv);
//ALLOC_ZVAL定义在zend_alloc.h
#define ALLOC_ZVAL(z) \
ZEND_FAST_ALLOC(z, zval, ZVAL_CACHE_LIST)
//ZEND_FAST_ALLOC定义在zend_alloc.h
#define ZEND_FAST_ALLOC(p, type, fc_type) \
(p) = (type *) emalloc(sizeof(type))
// INIT_PZVAL定义在 zend.h
#define INIT_PZVAL(z) \
(z)->refcount__gc = 1; \
(z)->is_ref__gc = 0;
SEPARATE_ZVAL()
这个宏是用于变量分离操作的。为什么要进行变量分离呢?主要原因还是在PHP语言里如果变量不是以引用的方式传递,那么在扩展里修改参数的值时是不能影响到原来的值的。但是PHP又是一个写时拷贝的。所以能在对一个由用户传递过来的参数进行修改时,必须进行变量分离,否则就会造成用户传递的参数被修改。
// SEPARATE_ZVAL定义在zend.h
#define SEPARATE_ZVAL(ppzv) \
{ \
zval *orig_ptr = *(ppzv); \
\
if (Z_REFCOUNT_P(orig_ptr) > 1) { \
Z_DELREF_P(orig_ptr); \
ALLOC_ZVAL(*(ppzv)); \
**(ppzv) = *orig_ptr; \
zval_copy_ctor(*(ppzv)); \
Z_SET_REFCOUNT_PP(ppzv, 1); \
Z_UNSET_ISREF_PP((ppzv)); \
} \
}
zval_copy_ctor()
zval 的拷贝构造函数(其实也是一个宏函数),通过这个宏,我们深拷贝一个zval。实例代码:
zval **old, *new;
*new = **old;
zval_copy_ctor(new);
// zval_copy_ctor定义在zend_variables.h
#define zval_copy_ctor(zvalue) _zval_copy_ctor((zvalue) ZEND_FILE_LINE_CC)
ZEND_API void _zval_copy_ctor_func(zval *zvalue ZEND_FILE_LINE_DC);
static inline void _zval_copy_ctor(zval *zvalue ZEND_FILE_LINE_DC)
{
if (zvalue->type <= IS_BOOL) {
return;
}
_zval_copy_ctor_func(zvalue ZEND_FILE_LINE_RELAY_CC);
}
变量值的访问:(定义在zend_operators.h)
Long | Boolean | Double | String value | String length |
---|---|---|---|---|
Z_LVAL() | Z_BVAL() | Z_DVAL() | Z_STRVAL() | Z_STRLEN() |
Z_LVAL_P() | Z_BVAL_P() | Z_DVAL_P() | Z_STRVAL_P() | Z_STRLEN_P() |
Z_LVAL_PP() | Z_BVAL_PP() | Z_DVAL_PP() | Z_STRVAL_PP() | Z_STRLEN_PP() |
HashTable | Object | Object properties | Object class entry | Resource value |
Z_ARRVAL() | Z_OBJ() | Z_OBJPROP() | Z_OBJCE() | Z_RESVAL() |
Z_ARRVAL_P() | Z_OBJ_P() | Z_OBJPROP_P() | Z_OBJCE_P() | Z_RESVAL_P() |
Z_ARRVAL_PP() | Z_OBJ_PP() | Z_OBJPROP_PP() | Z_OBJCE_PP() | Z_RESVAL_PP() |
获取量的类型:
// 定义在zend_operators.h 有关zvl的大部分操作宏都在这里文件里
#define Z_TYPE(zval) (zval).type
#define Z_TYPE_P(zval_p) Z_TYPE(*zval_p)
#define Z_TYPE_PP(zval_pp) Z_TYPE(**zval_pp)
PHP里预定义的变量类型有一下几种:
// 以下变量类型定义在zend.h里
/* data types */
/* All data types <= IS_BOOL have their constructor/destructors skipped */
#define IS_NULL 0
#define IS_LONG 1
#define IS_DOUBLE 2
#define IS_BOOL 3
#define IS_ARRAY 4
#define IS_OBJECT 5
#define IS_STRING 6
#define IS_RESOURCE 7
#define IS_CONSTANT 8
#define IS_CONSTANT_ARRAY 9