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中的时间处理
Oct 09 PHP
谈谈PHP语法(2)
Oct 09 PHP
php strcmp使用说明
Apr 22 PHP
php disk_free_space 返回目录可用空间
May 10 PHP
获取PHP警告错误信息的解决方法
Jun 03 PHP
php 判断字符串中是否包含html标签
Feb 17 PHP
使用图灵api创建微信聊天机器人
Jul 23 PHP
超强多功能php绿色集成环境详解
Jan 25 PHP
浅谈Yii乐观锁的使用及原理
Jul 25 PHP
php中curl和soap方式请求服务超时问题的解决
Jun 11 PHP
php设计模式之代理模式分析【星际争霸游戏案例】
Mar 23 PHP
php引用传递
Apr 01 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与MongoDB简介|安全|M+PHP应用实例详解
2013/06/17 PHP
PHP CodeIgniter框架的工作原理研究
2015/03/30 PHP
PHP编写RESTful接口
2016/02/23 PHP
PHP实现的简单分页类及用法示例
2016/05/06 PHP
PHP正则匹配到2个字符串之间的内容方法
2018/12/24 PHP
静态页面的值传递(三部曲)
2006/09/25 Javascript
通过JAVASCRIPT读取ASP设定的COOKIE
2007/02/15 Javascript
通过jQuery打造支持汉字,拼音,英文快速定位查询的超级select插件
2010/06/18 Javascript
jQuery find和children方法使用
2011/01/31 Javascript
input输入框的自动匹配(原生代码)
2013/03/19 Javascript
js操作checkbox遇到的问题解决
2013/06/29 Javascript
Jquery通过Ajax访问XML数据的小例子
2013/11/18 Javascript
推荐6款基于jQuery实现图片效果插件
2014/12/07 Javascript
Nodejs如何搭建Web服务器
2016/03/28 NodeJs
js实现无缝循环滚动
2020/06/23 Javascript
angularjs中ng-attr的用法详解
2016/12/31 Javascript
详解Javascript几种跨域方式总结
2017/02/27 Javascript
js实现自动图片轮播代码
2017/03/22 Javascript
vue 2.1.3 实时显示当前时间,每秒更新的方法
2018/09/16 Javascript
VueCli3构建TS项目的方法步骤
2018/11/07 Javascript
python按照多个字符对字符串进行分割的方法
2015/03/17 Python
python轻松实现代码编码格式转换
2015/03/26 Python
Python中矩阵库Numpy基本操作详解
2017/11/21 Python
Python 内置函数进制转换的用法(十进制转二进制、八进制、十六进制)
2018/04/30 Python
Python3用tkinter和PIL实现看图工具
2018/06/21 Python
解决pycharm py文件运行后停止按钮变成了灰色的问题
2018/11/29 Python
Python生成指定数量的优惠码实操内容
2019/06/18 Python
python3实现单目标粒子群算法
2019/11/14 Python
Java程序员常见面试题
2015/07/16 面试题
物业品质提升方案
2014/06/08 职场文书
英语课外活动总结
2014/08/27 职场文书
农村党支部书记司法四风问题对照检查材料
2014/09/26 职场文书
教师个人自我评价
2015/03/04 职场文书
2016学校先进集体事迹材料
2016/02/29 职场文书
2019个人年度目标制定攻略!
2019/07/12 职场文书
vue组件冲突之引用另一个组件出现组件不显示的问题
2022/04/13 Vue.js