PHP7内核之Reference详解


Posted in PHP onMarch 14, 2019

问题

上一章说过引用(REFERENCE)在PHP5的时候是一个标志位, 而在PHP7以后我们把它变成了一种新的类型:IS_REFERNCE. 然而引用是一种很常见的应用, 所以这个变化带来了很多的变化, 也给我们在做PHP7开发的时候, 因为有的时候疏忽忘了处理这个类型, 而带来不少的bug.

最简单的情况, 就是在处理各种类型的时候, 从此以后我们要多考虑这种新的类型, 比如在PHP7中, 这样的代码形式就变得很常见了:

try_again:
swtich (Z_TYPE_P(zv)) {
   case IS_TRING:
   break;
   case IS_ARRAY:
   break;
  ...
   case IS_REFERENCE:
   zv = Z_REFVAL_P(zv); //解引用
   goto try_again;
   break;
}

如果大家自己写的扩展, 如果忘了考虑这种新的类型, 那么就会导致问题.

为什么?
那么既然这种新类型会带来这么多问题, 那么当时为什么要用把引用变成一种类型呢? 为什么不还是使用一个标志位呢?

一句话来说, 就是我们不得不这么做. -_#

前面说到, Hashtable直接存储的是zval, 这样在符号表中, 俩个zval如何共用一个数值呢? 对于字符串等复杂类型来说还好, 我们貌似可以在zend_refcounted结构中加入一个标志位来表明是引用来解决, 然而这个也会遇到Change On Write带来的复制, 但是我们知道在PHP7中, 一些类型是直接存储在zval中的, 比如IS_LONG, 但是引用类型是需要引用计数的, 那么对于一个是IS_LONG并且又是IS_REFERNCE的zval该如何表示呢?

为此, 我们创造了这个新的类型:

PHP7内核之Reference详解

 如图所示, 引用是一种新的类型:zend_reference, 对于IS_REFERNCE类型的zval, zval.value.ref是一个指向zend_reference的指针, 它包含了引用计数和一个zval, 具体的zval的值是存在zval.value.ref->val中的.

所以对于IS_LONG的引用来说, 就用一个类型是IS_REFERNCE的zval, 它指向一个zend_reference, 而这个zend_reference->val中是一个类型为IS_LONG的zval.

Change On Write
PHP采用引用计数来做简单的垃圾回收, 考虑如下的代码:

<?php
1. $val = "laruence";
2. $ref = &$val;
3. $copy = $val;
?>

$ref和$val是指向同一个zval的引用, 在PHP5的时候, 我们是通过一个引用计数为2, 并且引用标志位为1来表示这种情况, 当把$val复制给$copy(line 3)的时候, 我们发现$val是一个计数大于1的引用, 所以要产生Change on write, 也就是分离. 所以我们需要复制这个zval.

而在PHP7中, 情况就变得简单了很多, 首先在引用赋值给$ref(line 2)的时候, 生成一个IS_REFERNCE类型, 然后因为此时有俩个变量引用它所以zend_reference这个结构的引用计数zval.value.ref->gc.refcount为2.

再随后的赋值给$copy(line 3)的时候, 发现$val是一个引用, 于是让$copy指向的是zval.value.ref->val, 也就是字符串值为laruence的zval, 然后把zval的引用计数+1, 也就是zval.value.ref->val.value.str.gc.refcount为2. 并没有产生复制.

从而这就很好的解决了上一章所说的PHP5的那个经典的问题, 比如我们在PHP7下运行上一章的那个问题, 我们得到的结果是:

$ php-7.0/sapi/cli/php /tmp/1.php
Used 0.00021380008539
Used 0.00020173048281

 可见确实没有发生复制, 从而不会产生任何的性能问题.

以上所述是小编给大家介绍的PHP7内核之Reference详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

PHP 相关文章推荐
模拟OICQ的实现思路和核心程序(二)
Oct 09 PHP
Discuz! Passport 通行证整合
Mar 27 PHP
编写漂亮的代码 - 将后台程序与前端程序分开
Apr 23 PHP
php 购物车实例(申精)
May 11 PHP
Base64在线编码解码实现代码 演示与下载
Jan 08 PHP
浅谈php中include文件变量作用域
Jun 18 PHP
Smarty变量用法详解
May 11 PHP
PHP实现批量检测网站是否能够正常打开的方法
Aug 23 PHP
PHP使用finfo_file()函数检测上传图片类型的实现方法
Apr 18 PHP
Thinkphp 5.0实现微信企业付款到零钱
Sep 30 PHP
Yii框架中用response保存cookie,用request读取cookie的原理解析
Sep 04 PHP
PHP实现批量修改文件名的方法示例
Sep 18 PHP
掌握PHP垃圾回收机制详解
Mar 13 #PHP
浅谈php的TS和NTS的区别
Mar 13 #PHP
浅谈PHP各环境下的伪静态配置
Mar 13 #PHP
Laravel框架实现的使用smtp发送邮件功能示例
Mar 12 #PHP
Laravel事件监听器用法实例分析
Mar 12 #PHP
PHP添加PNG图片背景透明水印操作类定义与用法示例
Mar 12 #PHP
PHP DB 数据库连接类定义与用法示例
Mar 11 #PHP
You might like
PHP文件读写操作之文件写入代码
2011/01/13 PHP
PHP中使用cURL实现Get和Post请求的方法
2013/03/13 PHP
php中serialize序列化与json性能测试的示例分析
2013/04/27 PHP
开源php中文分词系统SCWS安装和使用实例
2014/04/11 PHP
Yii实现Command任务处理的方法详解
2016/07/14 PHP
PHP递归统计系统中代码行数
2019/09/19 PHP
prototype 学习笔记整理
2009/07/17 Javascript
JavaScript入门之对象与JSON详解
2011/10/21 Javascript
JSON.stringify 语法实例讲解
2012/03/14 Javascript
Javascript对象属性方法汇总
2013/11/21 Javascript
JavaScript文档碎片操作实例分析
2015/12/12 Javascript
angularjs在ng-repeat中使用ng-model遇到的问题
2016/01/21 Javascript
浅析Bootstrap验证控件的使用
2016/06/23 Javascript
微信小程序 视图层(xx.xml)和逻辑层(xx.js)详细介绍
2016/10/13 Javascript
js实现界面向原生界面发消息并跳转功能
2016/11/22 Javascript
jQuery超简单遮罩层实现方法示例
2018/09/06 jQuery
详解Eslint 配置及规则说明
2018/09/10 Javascript
3分钟了解vue数据劫持的原理实现
2019/05/01 Javascript
vue-router 中 meta的用法详解
2019/11/01 Javascript
vant中的toast轻提示实现代码
2020/11/04 Javascript
python中执行shell命令的几个方法小结
2014/09/18 Python
Python爬虫实现百度图片自动下载
2018/02/04 Python
python OpenCV学习笔记之绘制直方图的方法
2018/02/08 Python
Python实现的基于优先等级分配糖果问题算法示例
2018/04/25 Python
pygame实现俄罗斯方块游戏(基础篇3)
2019/10/29 Python
python requests抓取one推送文字和图片代码实例
2019/11/04 Python
Python基于yield遍历多个可迭代对象
2020/03/12 Python
使用Python FastAPI构建Web服务的实现
2020/06/08 Python
python爬虫多次请求超时的几种重试方法(6种)
2020/12/01 Python
重新定义牛仔布,100美元以下:Warp + Weft
2018/07/25 全球购物
励志演讲稿范文
2014/04/29 职场文书
考核评语大全
2014/04/29 职场文书
百日安全生产活动总结
2014/07/05 职场文书
婚礼女方父母答谢词
2015/01/04 职场文书
2015年团队工作总结范文
2015/05/04 职场文书
团支部组织委员竞选稿
2015/11/21 职场文书