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 相关文章推荐
用PHP制作的意见反馈表源码
Mar 11 PHP
PHP静态新闻列表自动生成代码
Jun 14 PHP
PHP 获取远程文件内容的函数代码
Mar 24 PHP
php下通过IP获取地理位置的代码(小偷程序)
Jun 09 PHP
input file获得文件根目录简单实现
Apr 26 PHP
php中sprintf与printf函数用法区别解析
Feb 17 PHP
php中Session的生成机制、回收机制和存储机制探究
Aug 19 PHP
php实现mysql备份恢复分卷处理的方法
Dec 26 PHP
从wamp到xampp的升级之路
Apr 08 PHP
php简单实现多维数组排序的方法
Sep 30 PHP
PHP实现小偷程序实例
Oct 31 PHP
php引用和拷贝的区别知识点总结
Sep 23 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中获取远程客户端的真实ip地址的方法
2011/08/03 PHP
PHP排序算法的复习和总结
2012/02/15 PHP
php json_encode值中大括号与花括号区别
2013/09/30 PHP
PHP连接MySQL查询结果中文显示乱码解决方法
2013/10/25 PHP
PHP6 中可能会出现的新特性预览
2014/04/04 PHP
PHP防止表单重复提交的几种常用方法汇总
2014/08/19 PHP
Yii2.0高级框架数据库增删改查的一些操作
2015/11/16 PHP
js获取url参数代码实例分享(JS操作URL)
2013/12/13 Javascript
Js中使用hasOwnProperty方法检索ajax响应对象的例子
2014/12/08 Javascript
Javascript中的call()方法介绍
2015/03/15 Javascript
JavaScript中的some()方法使用详解
2015/06/09 Javascript
Bootstrap使用基础教程详解
2016/09/05 Javascript
微信小程序 石头剪刀布实例代码
2017/01/04 Javascript
js 将input框中的输入自动转化成半角大写(税号输入框)
2017/02/16 Javascript
从零开始学习Node.js系列教程三:图片上传和显示方法示例
2017/04/13 Javascript
简单快速的实现js计算器功能
2017/08/17 Javascript
ReactNative之FlatList的具体使用方法
2017/11/29 Javascript
vue导出html、word和pdf的实现代码
2018/07/31 Javascript
iview的table组件自带的过滤器实现
2019/07/12 Javascript
jquery 插件重新绑定的处理方法分析
2019/11/23 jQuery
微信小程序实现抖音播放效果的实例代码
2020/04/11 Javascript
探索node之事件循环的实现
2020/10/30 Javascript
[01:04:32]DOTA2-DPC中国联赛 正赛 Aster vs LBZS BO3 第二场 2月23日
2021/03/11 DOTA
Python2.5/2.6实用教程 入门基础篇
2009/11/29 Python
Python将list中的string批量转化成int/float的方法
2018/06/26 Python
Numpy 改变数组维度的几种方法小结
2018/08/02 Python
浅谈Pytorch中的torch.gather函数的含义
2019/08/18 Python
python识别文字(基于tesseract)代码实例
2019/08/24 Python
pandas实现将日期转换成timestamp
2019/12/07 Python
在Matplotlib图中插入LaTex公式实例
2020/04/17 Python
Python类class参数self原理解析
2020/11/19 Python
涉外经济法专业毕业生推荐信
2013/11/24 职场文书
《燕子》教学反思
2014/02/18 职场文书
行政办公室岗位职责
2014/03/18 职场文书
安全生产目标管理责任书
2014/07/25 职场文书
学校清洁工岗位职责
2015/04/15 职场文书