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

浅析PHP原理之变量(Variables inside PHP)

程序员文章站 2022-07-20 08:26:00
或许你知道,或许你不知道,php是一个弱类型,动态的脚本语言。所谓弱类型,就是说php并不严格验证变量类型(严格来讲,php是一个中强类型语言,这部分内容会在以后的文章中叙...

或许你知道,或许你不知道,php是一个弱类型,动态的脚本语言。所谓弱类型,就是说php并不严格验证变量类型(严格来讲,php是一个中强类型语言,这部分内容会在以后的文章中叙述),在申明一个变量的时候,并不需要显示指明它保存的数据的类型:

复制代码 代码如下:

<?php
  $var = 1; //int
  $var = "laruence"; //string
  $var = 1.0002; //float
  $var = array(); // array
  $var = new exception('error'); //object;

动态语言,就是说,php的语言结构在运行期是可以改变的,比如我们在运行期require一个函数定义文件,从而导致语言的函数表动态的改变。

所谓脚本语言,就是说,php并不是独立运行的,要运行php我们需要php解析器:

复制代码 代码如下:

  /usr/bin/php -f example.ph

我前面的文章中已经讲过,php的执行是通过zend engine(ze, zend引擎), ze是用c编写的,大家都知道c是一个强类型语言,也就是说,在c中所有的变量在它被声明到最终销毁,都只能保存一种类型的数据。 那么php是如何在ze的基础上实现弱类型的呢?

在php中,所有的变量都是用一个结构-zval来保存的, 在zend/zend.h中我们可以看到zval的定义:

复制代码 代码如下:

  typedef struct _zval_struct {
    zvalue_value value;
    zend_uint refcount;
    zend_uchar type;
    zend_uchar is_ref;
  } zval;
 
其中zvalue_value是真正保存数据的关键部分,现在到了揭晓谜底的时候了,php是如何在ze的基础上实现弱类型的呢? 因为zvalue_value是个联合体(union),
复制代码 代码如下:

typedef union _zvalue_value {
    long lval;
    double dval;
    struct {
        char *val;
        int len;
    } str;
    hashtable *ht;
    zend_object_value obj;
} zvalue_value;

那么这个结构是如何储存php中的多种类型的呢?

php中常见的变量类型有:
1. 整型/浮点/长整型/bool值 等等
2. 字符串
3. 数组/关联数组
4. 对象
5. 资源

php根据zval中的type字段来储存一个变量的真正类型,然后根据type来选择如何获取zvalue_value的值,比如对于整型和bool值:

复制代码 代码如下:

   zval.type = is_long;//整形
   zval.type = is_bool;//布尔

就去取zval.value.lval,对于bool值来说lval∈(0|1);
如果是双精度,或者float则会去取zval.value的dval。
而如果是字符串,那么:
复制代码 代码如下:

   zval.type = is_strin

这个时候,就会取:
zval.value.str
而这个也是个结构,存有c分格的字符串和字符串的长度。
而对于数组和对象,则type分别对应is_array, is_object, 相对应的则分别取zval.value.ht和obj
比较特别的是资源,在php中,资源是个很特别的变量,任何不属于php内建的变量类型的变量,都会被看作成资源来进行保存,比如,数据库句柄,打开的文件句柄等等。 对于资源:
复制代码 代码如下:

   type = is_resourc

这个时候,会去取zval.value.lval, 此时的lval是个整型的指示器, 然后php会再根据这个指示器在php内建的一个资源列表中查询相对应的资源(这部分的内容,我以后会单独开一个篇文章来介绍),目前,你只要知道此时的lval就好像是对应于资源链表的偏移值。
复制代码 代码如下:

 zend_fetch_resource(con, type, zval *, default, resource_name, resource_type);

借用这样的机制,php就实现了弱类型,因为对于ze的来说,它所面对的永远都是同一种类型,那就是zval。