PHP代码如下:

$php_var = 1;

对应C的代码是:

zval* c_var;    //定义PHP变量指针

MAKE_STD_ZVAL(c_var);  //初始化PHP变量

ZVAL_LONG(c_var,1) ;//赋值

ZEND_SET_SYMBL( EG(active_symbol_table), " php_var ", c_var);//注册到全局变量符号表

一.首先看第一行: zval* c_var;//申明一个zval指针c_var; zval的结构如下:

struct _zval_struct {

/* Variable information */

zvalue_value value;     /* 变量的值 */

zend_uint refcount;     /* 引用计数,垃圾回收的时候用到 */

zend_uchar type;        /* 变量类型 */

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

};

typedef struct _zval_struct zval;

其中值zvalue_value的结构如下:

typedef union _zvalue_value {

long lval;              /* 长整形*/

double dval;            /* 双精度类型 */

struct {                  /* 字符串类型的值 */

char *val;

int len;

} str;

HashTable *ht;              /* 数组类型的值 */

zend_object_value obj;     /*对象类型的值*/

} zvalue_value;

二.接下来看第二行: MAKE_STD_ZVAL(new_val);//变量初始化 相关宏如下: //初始化

#define MAKE_STD_ZVAL(zv)                \

ALLOC_ZVAL(zv); \

INIT_PZVAL(zv);

#define ALLOC_ZVAL(z)   \

ZEND_FAST_ALLOC(z, zval, ZVAL_CACHE_LIST)

#define ZEND_FAST_ALLOC(p, type, fc_type)   \

(p) = (type *) emalloc(sizeof(type))

#define INIT_PZVAL(z)       \

(z)->refcount = 1;      \

(z)->is_ref = 0;

展开后为:

(c_var) = (zval *) emalloc(sizeof(zval));  //分配内存

(c_var)-> refcount = 1;  //引用计数初始化

(c_var)-> is_ref = 0; //是否引用

可以看到其作用就是分配内存,初始化refcount,is_ref

三.下面看第三行 ZVAL_LONG(c_var,1) 相关宏为:

//定义值

#define ZVAL_LONG(z, l) {           \

Z_TYPE_P(z) = IS_LONG;      \

Z_LVAL_P(z) = l;            \

}

#define Z_TYPE_P(zval_p)    Z_TYPE(*zval_p)

#define Z_TYPE(zval)        (zval).type

#define Z_LVAL_P(zval_p)    Z_LVAL(*zval_p)

#define Z_LVAL(zval)            (zval).value.lval

展开后为:

(* c_var).type = IS_LONG;

(* c_var).value = 1;

四:接下来看第四行: ZEND_SET_SYMBOL( EG(active_symbol_table), “php_var”, c_var); 首先说明下PHP的变量是存在一个hashtable里的

struct _zend_executor_globals {

….

HashTable symbol_table;//全局变量的符号表

HashTable *active_symbol_table;//局部变量的符号表

…..

};

Hashtable的Key为变量的名称,即php_var,值为指向PHP变量的指针,即c_var指针; 相关宏为:

#define ZEND_SET_SYMBOL(symtable, name, var)          \   {                                                     \

char *_name = (name);                         \

ZEND_SET_SYMBOL_WITH_LENGTH(symtable, _name, strlen(_name)+1, var, 1, 0);   \

}

//主要的实现为下面这个函数:

#define ZEND_SET_SYMBOL_WITH_LENGTH(symtable, name, name_length, var, _refcount, _is_ref)                                                       \

{

zval **orig_var;                                        \

if (zend_hash_find(symtable, (name), (name_length), (void **) &orig_var)==SUCCESS                                                         \

&& PZVAL_IS_REF(*orig_var)) {                     \

(var)->refcount = (*orig_var)->refcount;                  \

(var)->is_ref = 1;                                \

if (_refcount) {                                      \

(var)->refcount += _refcount-1;               \

}                                             \

zval_dtor(*orig_var);                             \

**orig_var = *(var);                                  \

FREE_ZVAL(var);                               \

} else {                                              \

(var)->is_ref = _is_ref;                              \

if (_refcount) {                                      \

(var)->refcount = _refcount;                      \

}                                             \

zend_hash_update(symtable, (name), (name_length), &(var), sizeof(zval *), NULL);                                                           \

}                                                  \

}

该函数的功能是:

1. 如果全局符号表已经存在该变量且是引用类型,则

a. 将原来变量的引用计数refcount,is_ref信息赋给c_var;

b. 释放掉原来变量zvalue的值,比如原来其值指向的是一个mysql连接资源,则释放该资源。

c. 将c_var指向的变量赋值给原来的变量 d. 释放c_var的内存空间 这样保证了,如果变量被应用,值一起改变。比如如果前面有$b=&a;

2. 如果全局符号表不存在该变量或者存在该变量但不是引用变量,则直接改变其值。

php从内存中获取源码_PHP源码分析之变量的存储过程分解相关推荐

  1. php从内存中获取源码_【PHP7源码分析】PHP内存管理

    作者: 顺风车运营研发团队 李乐 第一章 从操作系统内存管理说起 程序是代码和数据的集合,进程是运行着的程序:操作系统需要为进程分配内存:进程运行完毕需要释放内存:内存管理就是内存的分配和释放: 1. ...

  2. gpg 中标麒麟获取 密钥失败_PHP实现码云Gitee的WebHook密钥验证算法

    码云的 WebHook 支持两种验证方式,一种是明文密码验证,另外一种是密钥验证 用户通过配置不公开的 WebHook 密钥,在请求时对请求内容签名,服务端在收到请求后以同样的密钥进行签名验证,以确认 ...

  3. A13在c语言中是合法变量吗,C语言中局部变量和全局变量等在内存中的存放位置.doc-资源下载在线文库www.lddoc.cn...

    C语言中局部变量和全局变量_等在内存中的存放位置.doc C 语言中局部变量和全局变量 变量的存储类别static,extern,auto,register 8.8 局部变量和全局变量在讨论函数的形参 ...

  4. C#中获取当前应用程序的路径及环境变量

    2019独角兽企业重金招聘Python工程师标准>>> C#获取当前应用程序所在路径及环境变量 一.获取当前文件的路径 string str1=Process.GetCurrentP ...

  5. java中的onresume_android onCreate onResume中获取 View 宽高为0分析

    1.问题测试 xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_par ...

  6. 用wamper打开php源码_php源码该怎么设置数据库

    php源码该怎么设置数据库 php源码修改数据库的设置方法:首先在源码中找到程序的配置文件:然后用文本编辑器打开:之后修改数据库地址.数据库名.用户名.连接密码.数据库前缀等参数:最后保存即可.这里以 ...

  7. 如何直接从手机内存中获取CellID?

    前几天在研究通过RIL_GetCellTowerInfo的API获取CellID的问题,当时发现了一些机型没法直接通过这个API来获取CellID的问题,在网上查了一些资料如何使用C#在Windows ...

  8. 内存中的fas和lsp

    写一个小程序: (defun C:TTT() (SETQ A 8)(PRINC "TTT")) (defun C:AAA() (PRINC "AA")) (de ...

  9. spring 源码_spring源码系列(一)——spring循环引用

    众所周知spring在默认单例的情况下是支持循环引用的 为了节省图片大小我把那些可以动得gif图片做成了只循环一次,如果看到图片不动了请右键选择在新标签打开,那么图片就会动,手机用户则更简单,直接手指 ...

最新文章

  1. 使用相对路径时,./、../、../../,代表的什么?
  2. 李伯虎院士:新一代人工智能引领下的智造制造初步实践
  3. foxmail提示不知道这样的主机_不知道和婚礼策划师沟通时谈哪些?这样做让你高效备婚...
  4. WPF converter(包含传递复杂参数)
  5. 深度学习——05、深度学习框架Caffe
  6. http://www.cnblogs.com/ITtangtang/archive/2012/05/21/2511749.html
  7. Mysql数据库名和表名的大小写敏感性问题
  8. zookeeper系列(三)zookeeper的使用--开源客户端
  9. newtonsoft
  10. C语言-输入任意多个数字,数字间用空格隔开,然后将数字求和。
  11. 一键开关机电路图解析
  12. 某站的视频下载与合并,m4s的处理。
  13. PTA题目 : 用指针实现将一个3*3的整型矩阵转置
  14. 微信公众号js接口安全域名的MP_verify_*.txt文件的放置路径
  15. 7-62 贴“福”字
  16. java设置窗体不透明的函数_透明窗口(窗口上面文字图片等内容不透明)的实现(使用SetLayeredWindowAttributes API函数)...
  17. 2021-02-03-linux命令
  18. 非线性方程的数值解法:牛顿下山法 python
  19. 2022年「博客之星」 无知的人_的程序人生
  20. ocmt:高维固定效应模型的变量筛选问题

热门文章

  1. 《阿里巴巴Android开发手册》v1.0.1更新,优化部分内容和示例代码
  2. 【产品活动】阿里云GPU云服务器年付5折!阿里云异构计算助推行业发展!
  3. 在 Kubernetes 集群中使用 MetalLB 作为 Load Balancer(上)
  4. 从代码到 Docker、Kubernetes、Istio、Knative……,或许是时候重新思考从代码到云的编程了...
  5. 探秘HDFS —— 发展历史、核心概念、架构、工作机制 (上)| 博文精选
  6. 格罗方德起诉台积电侵犯16项专利、影响巨大;中兴通讯与印尼Smartfren展开合作;网传FB开发新通讯应用Threads……...
  7. java 最大矩形_Java实现 LeetCode 85 最大矩形
  8. 计算机图形学有序边表作业,《计算机图形学》有序边表填充算法.docx
  9. jre for mac 删除_在 Mac 的 Docker Desktop 中运行 K8s
  10. Zipkin 存储追踪数据至 MySQL