欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

PHP源码分析-弱类型变量实现

程序员文章站 2022-03-19 23:48:49
php是弱类型,动态的语言脚本。在申明一个变量的时候,并不需要指明它保存的数据类型。 例如:

php是弱类型,动态的语言脚本。在申明一个变量的时候,并不需要指明它保存的数据类型。

例如:

<?php 
$var = 1; 
$var = "variable"; 
$var = 1.00; 
$var = array(); 
$var = new object(); 
<?php
$var = 1;
$var = "variable";
$var = 1.00;
$var = array();
$var = new object();
动态变量,在运行期间是可以改变的,并且在使用前无需声明变量类型。

 


那么,问题一、zend引擎是如何用c实现这种弱类型的呢?
实际上,在php中声明的变量,在ze中都是用结构体zval来保存的。

首先我们打开zend/zend.h来看zval的定义:


typedef struct _zval_struct zval; 
 
struct _zval_struct { 
    /* variable information */ 
    zvalue_value value;     /* value */ 
    zend_uint refcount__gc; 
    zend_uchar type;    /* active type */ 
    zend_uchar is_ref__gc; 
}; 
 
typedef union _zvalue_value { 
    long lval;  /* long value */ 
    double dval;    /* double value */ 
    struct { 
        char *val; 
        int len; 
    } str; 
    hashtable *ht;  /* hash table value */ 
    zend_object_value obj; 
} zvalue_value; 
typedef struct _zval_struct zval;

struct _zval_struct {
 /* variable information */
 zvalue_value value;  /* value */
 zend_uint refcount__gc;
 zend_uchar type; /* active type */
 zend_uchar is_ref__gc;
};

typedef union _zvalue_value {
 long lval; /* long value */
 double dval; /* double value */
 struct {
  char *val;
  int len;
 } str;
 hashtable *ht; /* hash table value */
 zend_object_value obj;
} zvalue_value;
zend/zend_types.h:


typedef unsigned char zend_bool; 
typedef unsigned char zend_uchar; 
typedef unsigned int zend_uint; 
typedef unsigned long zend_ulong; 
typedef unsigned short zend_ushort; 
typedef unsigned char zend_bool;
typedef unsigned char zend_uchar;
typedef unsigned int zend_uint;
typedef unsigned long zend_ulong;
typedef unsigned short zend_ushort;
从上述代码中,可以看到_zvalue_value是真正保存数据的关键部分。通过共用体实现的弱类型变量声明

 

 

问题二、zend引擎是如何判别、存储php中的多种数据类型的呢?
_zval_struct.type中存储着一个变量的真正类型,根据type来选择如何获取zvalue_value的值。

type值列表(zend/zend.h): 
#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 
type值列表(zend/zend.h):
#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
来看一个简单的例子:
<?php 
    $a = 1; 
    //此时zval.type = is_long,那么zval.value就去取lval.  
    $a = array(); 
    //此时zval.type = is_array,那么zval.value就去取ht. 
<?php
 $a = 1;
 //此时zval.type = is_long,那么zval.value就去取lval.
 $a = array();
 //此时zval.type = is_array,那么zval.value就去取ht.
这其中最复杂的,并且在开发第三方扩展中经常需要用到的是"资源类型".
在php中,任何不属于php的内建的变量类型的变量,都会被看作资源来进行保存。
比如:句柄、打开的文件句柄、打开的socket句柄。

资源类型,需要使用ze提供的api函数来注册,资源变量的声明和使用将在单独的篇目中进行详细介绍。

正是因为ze这样的处理方式,使php就实现了弱类型,而对于ze的来说,它所面对的永远都是同一种类型zval

 

摘自 god's blog