前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >深入理解PHP原理之变量分离/引用(Variables Separation)

深入理解PHP原理之变量分离/引用(Variables Separation)

作者头像
Java架构师必看
发布2021-03-22 10:32:05
1.4K0
发布2021-03-22 10:32:05
举报
文章被收录于专栏:Java架构师必看

引自:http://www.laruence.com/[风雪之隅]在前面的文章中我已经介绍了P

引自: http://www.laruence.com/ [风雪之隅 ]

在前面的文章中我已经介绍了PHP的变量的内部表示(深入理解PHP原理之变量(Variables inside PHP)),以及PHP中作用域的实现机制(深入理解PHP原理之变量作用域(Scope inside PHP))。这节我们就接着前面的文章,继续介绍PHP中变量分离和引用的概念:

首先我们回顾一下zval的结构:

struct _zval_struct {         /* Variable information */         zvalue_value value;             /* value */         zend_uint refcount;         zend_uchar type;        /* active type */         zend_uchar is_ref; }; 其中的refcount和is_ref字段我们一直都没有介绍过,我们知道PHP是一个长时间运行的服务器端的脚本解释器。那么对于它来说,效率和资源占用率是一个很重要的衡量标准,也就是说,PHP必须尽量介绍内存占用率,考虑下面这段代码:

var = "laruence";   

这样的代码在我们平时的脚本中是很常见的,如果PHP对于每一个变量赋值都重新分配内存,copy数据的话,那么上面的这段代码公要申请18个字节的内存空间,而我们也很容易的看出来,上面的代码其实根本没有必要申请俩份空间,呵呵,PHP的开发者也看出来了:

我们之前讲过,PHP中的变量是用一个存储在symbol_table中的符号名,对应一个zval来实现的,比如对于上面的第一行代码,会在symbol_table中存储一个值”var”, 对应的有一个指针指向一个zval结构,变量值”laruence”保存在这个zval中,所以不难想象,对于上面的代码来说,我们完全可以让”var”和”var_dup”对应的指针都指向同一个zval就可以了。

PHP也是这样做的,这个时候就需要介绍我们之前一直没有介绍过的zval结构中的refcount字段了。 refcount,顾名思义,记录了当前的zval被引用的计数。 比如对于代码:

var = 1;    var_dup = var; ?> 第一行,创建了一个整形变量,变量值是1。 此时保存整形1的这个zval的refcount为1。 第二行,创建了一个新的整形变量,变量也指向刚才创建的zval,并将这个zval的refcount加1,此时这个zval的refcount为2。 PHP提供了一个函数可以帮助我们了解这个过程debug_zval_dump:

var = 1; debug_zval_dump(var);var_dup =

long(1) refcount(2) long(1) refcount(3) 如果你奇怪 ,var的refcount应该是1啊? 我们知道,对于简单变量,PHP是以传值的形式穿参数的。也就是说,当执行debug_zval_dump(var)的时候,var会以传值的方式传递给debug_zval_dump,也就是会导致var的refcount加1,所以我们只要能看到,当变量赋值给一个变量以后,能导致zval的refcount加1这个事实即可。

现在我们回头看文章开头的代码, 当执行了最后一行unset($var)以后,会发生什么呢? 对,既是refcount减1,上代码:

var = "laruence";   

string(8) "laruence" refcount(2) 但是,对于下面的代码呢?

var = "laruence";    var_dup的值应该还是”laruence”, 那么这又是怎么实现的呢? 这就是PHP的copy on write机制: PHP在修改一个变量以前,会首先查看这个变量的refcount,如果refcount大于1,PHP就会执行一个分离的例程, 对于上面的代码,当执行到第三行的时候,PHP发现var指向的zval的refcount大于1,那么PHP就会复制一个新的zval出来,将原zval的refcount减1,并修改symbol_table,使得var和

上代码测试:

var = "laruence";   

long(1) refcount(2) string(8) "laruence" refcount(2) 现在我们知道,当使用变量复制的时候 ,PHP内部并不是真正的复制,而是采用指向相同的结构来尽量节约开销。那么,对于PHP中的引用,那又是如何实现呢?

var = "laruence";   

if((*val)->is_ref || (*val)->refcount<2){         //不执行Separation         ... ;//process } 但是,问题又来了,对于如下的代码,又会怎样呢?

var = "laruence";    var_ref = &var和var_dup, 又有一对change on write机制的变量对var和var_ref,这个情况又是如何运作的呢?

当第二行执行的时候,和前面讲过的一样,var_dup 和 var 指向相同的zval, refcount为2. 当执行第三行的时候,PHP发现要操作的zval的refcount大于1,则,PHP会执行Separation, 将var_dup分离出去,并将var和

基于这样的分析,我们就可以让debug_zval_dump出refcount为1的结果来:

var = "laruence";    

string(8) "laruence" refcount(1) 详细原因,读者你只要稍加分析就能得出,我就不越俎代庖了。;)

这次我们介绍了PHP的变量分离机制,下次我会继续介绍如果在扩展中接收和传出PHP脚本中的参数。另外,因为最近变动比较大(换工作),所以抱歉这么长时间才有更新

本文由来源 21aspnet,由 javajgs_com 整理编辑,其版权均为 21aspnet 所有,文章内容系作者个人观点,不代表 Java架构师必看 对观点赞同或支持。如需转载,请注明文章来源。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档