CSDN博客

img taft

Zend变量探索

发表于2003/3/31 9:30:00  1766人阅读

Zend变量探索

 

V2.0

 

taft@wjl.cn

 

最后更新 2006-1-27

 

[摘要] 本文介绍了作为PHP语言核心的Zend引擎对处理脚本变量的机制。

 

写过CC++或者是java代码肯定清楚,在使用变量的时候必须先定义一个变量,比如整型int、字符型char等等,这样在编译的时候编译器就会自动给变量划分相应的储存空间。先定义(有的还必须初始化)后使用是编译型语言的基本原则,否则编译的时候不能通过。解释型语言就没有这么“麻烦”了,想用什么变量按照命名规则使用就是了,不用声明。这的确很方便,特别适合初学计算机程序设计,如BASIC语言就是一个地道的解释型语言。大家可能会注意到,麻烦吗?是的,在解释型语言中变量的出生到灭亡都麻烦,做的工作不比编译型语言少。开放原代码的PHP同样是解释型语言。

 

1、   变量在Zend中的定义:

 

Zend的变量分内部变量(internal)和外部变量(external),内部变量主要是完成引擎动作或各扩展模块的“内部员工”,定义和c语言完全一样。外部变量是Zend引擎扫描到的PHP脚本文件里使用的变量和PHP函数的一部分形参变量,定义为zval。请看下面的定义:

 

typedef  struct _zval_struct   zval;

typedef  struct _zend_class_entry   zend_class_entry;

 

typedef union _zvalue_value {

    long lval;                  /* long value */

    double dval;                 /* double value */

    struct {

       char *val;

       int len;

    } str;

    HashTable *ht;              /* hash table value */

    struct {

       zend_class_entry *ce;

       HashTable *properties;

    } obj;

} zvalue_value;

 

struct _zval_struct {

    /* Variable information */

    zvalue_value value;      /* value */

    zend_uchar type;  /* active type  3*/

    zend_uchar is_ref;  /*是否为引用*/

    zend_ushort refcount;

};

zend.h

 

Zend用一个容器来处理所有的外部变量,这个容器就是上面定义的联合体(union_zvalue_value,它包含了常见的长整型(long),浮点双精度型(double),字符串型(结构体,分别有字符串首地址、字符串长度),Hash表,类结构体(如果你定义一个类,这个结构体就起作用了)。通常,在C++ 中容器被定义为一个以保存一批对象为主要用途的类。Zend变量容器的作用显然和C++有类似之处,根据用户不同的赋值,容器呈现不同的类型。Zend把外部变量封装在结构体_zal_struct里面,此结构体储存了变量的基本信息:变量值(value)、变量类型(type)、是否为引用(PHP4以上)、引用计数(PHP4以上)。定义一个外部变量如下:

     zval  my_zval, *pmy_zval;

引用一个变量的值:my_zval.value;取得变量类型:my_zval.type

还要提一点的是,这里的typeZend所定义的,而不是C语言里的原子类型。列出来如下:

 

/* data types */

#define IS_NULL                 0

#define IS_LONG                 1

#define IS_DOUBLE               2

#define IS_STRING               3

#define IS_ARRAY                4

#define IS_OBJECT               5

#define IS_BOOL                 6

#define IS_RESOURCE             7

#define IS_CONSTANT             8

#define IS_CONSTANT_ARRAY       9

zend.h 318行)

这些类型不用于定义变量,主要用作类型的判断。

下图更清楚地反映了相互的关系:

struct _zval_struct {

/* Variable information    zvalue_value value;      zend_uchar type; 

    zend_uchar is_ref;      zend_ushort refcount;

};

 

typedef union _zvalue_value {

    long lval;                   /* long value */

double dval;                 /* double value */

    struct {

       char *val;

       int len;

    } str;

    HashTable *ht;               /* hash table value */

    struct {

       zend_class_entry *ce;

       HashTable *properties;

    } obj;

} zvalue_value;

 

IS_NULL           0

IS_LONG           1

IS_DOUBLE         2

IS_STRING         3

IS_ARRAY          4

IS_OBJECT         5

IS_BOOL           6

IS_RESOURCE       7

IS_CONSTANT       8

IS_CONSTANT_ARRAY 9

 

 

 

2、变量在Zend中的创建

 

Zend扫描解释到外部变量并用zval定义之后,必须在内存中为此变量申请合适的空间,然后把变量添加到符号表中去,如果是全局变量,则添加到全局符号表中。首先Zend用到了一个宏:MAKE_STD_ZVAL

 

#define MAKE_STD_ZVAL(zv)               /   //分配新的ZVAL容器

    ALLOC_ZVAL(zv); /

    INIT_PZVAL(zv);

zend.h

 

此宏先调用ALLOC_ZVAL在内存中分配合适的空间,然后用INIT_PZVAL(zv)初始化变量:置容器引用计数(refcount)为1,是否为引用is_ref为否(0),

Zend的下一步工作是把变量添加到符号表里去,使用了宏 ZEND_SET_SYMBOL(symtable, name, var),它先检查这个值是否已经存在于符号表中,如果存在,将新符号转变为一个引用(reference)变量(并自动释放旧的zval容器)。

 

3、举 

 

最后在这里给出几个变量诞生的全过程:

假设用户在php脚本文件有个变量$new_long_name=100

Zend引擎的处理如下:

 

zval  *new_long;              //定义一个容器变量

MAKT_STD_ZVAL(new_long);      //为新变量开辟内存空间并初始化

new_long->type = IS_LONG;

new_long->value.lval = 100;

ZEND_SET_SYMBOL(EG(active_symbol_table),”new_long_name”,new_long);    //添加变量到符号表

 

ZEND_SET_SYMBOL使用EG宏存取Zend执行器全局变量。通过指定EG(active_symbol_table),可以访问当前活动符号表,处理活动且局部范围内的符号。根据函数调用的不同局部范围也会变化。

字符串的创建:

 

zval  *new_string;

char  *string_contents = “This is a new string variable”;

MAKE_STD_ZVAL(new_string);

new_string->type = IS_STRING;

new_string->value.str.len = strlen(string_contents);

new_string->value.str.val= estrdup(string_contents);

 

 

0 0

相关博文

我的热门文章

img
取 消
img