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代码
Aug 08 PHP
php中替换字符串中的空格为逗号','的方法
Jun 09 PHP
phpmailer发送邮件之后,返回收件人是否阅读了邮件的方法
Jul 19 PHP
PHP使用glob函数遍历目录或文件夹的方法
Dec 16 PHP
PHP框架Laravel的小技巧两则
Feb 10 PHP
php中header跳转使用include包含解决参数丢失问题
May 08 PHP
ThinkPHP数据操作方法总结
Sep 28 PHP
PHP 中 Orientation 属性判断上传图片是否需要旋转
Oct 16 PHP
PHP通过反射动态加载第三方类和获得类源码的实例
Nov 27 PHP
Zend Framework前端控制器用法示例
Dec 11 PHP
php把字符串指定字符分割成数组的方法
Mar 12 PHP
Centos7 Yum安装PHP7.2流程教程详解
Jul 02 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
PHP5 安装方法
2007/01/15 PHP
php中stdClass的用法分析
2015/02/27 PHP
PHP 实现重载
2021/03/09 PHP
javascript call方法使用说明
2010/01/11 Javascript
js 赋值包含单引号双引号问题的解决方法
2014/02/26 Javascript
node.js+Ajax实现获取HTTP服务器返回数据
2014/11/26 Javascript
轻松创建nodejs服务器(6):作出响应
2014/12/18 NodeJs
详解JavaScript中的forEach()方法的使用
2015/06/08 Javascript
JS实现网页右侧带动画效果的伸缩窗口代码
2015/10/29 Javascript
基于BootStrap Metronic开发框架经验小结【二】列表分页处理和插件JSTree的使用
2016/05/12 Javascript
jQuery模拟select实现下拉菜单功能
2016/06/20 Javascript
js实现tab切换效果
2017/02/16 Javascript
使用Node.js实现ORM的一种思路详解(图文)
2017/10/24 Javascript
js正则相关知识点专题
2018/05/10 Javascript
Vue使用vue-area-linkage实现地址三级联动效果的示例
2018/06/27 Javascript
JavaScript文本特效实例小结【3个示例】
2018/12/22 Javascript
vue slot与传参实例代码讲解
2019/04/28 Javascript
ElementUI之Message功能拓展详解
2019/10/18 Javascript
javascript操作元素的常见方法小结
2019/11/13 Javascript
vue中路由跳转不计入history的操作
2020/09/21 Javascript
JavaScript实现手风琴效果
2021/02/18 Javascript
Python3基础之输入和输出实例分析
2014/08/18 Python
Python实时获取cmd的输出
2015/12/13 Python
Fiddler如何抓取手机APP数据包
2016/01/22 Python
深入理解Python3中的http.client模块
2017/03/29 Python
简单了解python协程的相关知识
2019/08/31 Python
解决django后台管理界面添加中文内容乱码问题
2019/11/15 Python
Django框架配置mysql数据库实现过程
2020/04/22 Python
python opencv 实现读取、显示、写入图像的方法
2020/06/08 Python
HTML5 placeholder(空白提示)属性介绍
2013/08/07 HTML / CSS
丹麦优惠购物网站:PLUSSHOP
2019/03/24 全球购物
澳洲最大的时尚奢侈品电商平台:Cettire
2020/06/15 全球购物
高中毕业的自我鉴定
2013/12/09 职场文书
教师节祝酒词
2015/08/11 职场文书
element多个表单校验的实现
2021/05/27 Javascript
MySQL 如何限制一张表的记录数
2021/09/14 MySQL