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 cookie 登录验证示例代码
Mar 16 PHP
php中计算程序运行时间的类代码
Nov 03 PHP
使用Linux五年积累的一些经验技巧
Jun 20 PHP
PHP生成二维码的两个方法和实例
Jul 01 PHP
php强制运行广告的方法
Dec 01 PHP
thinkPHP模型初始化实例分析
Dec 03 PHP
thinkPHP多语言切换设置方法详解
Nov 11 PHP
phpstudy默认不支持64位php的解决方法
Feb 20 PHP
ajax调用返回php接口返回json数据的方法(必看篇)
May 05 PHP
PHP简单实现循环链表功能示例
Nov 10 PHP
PHP后台实现微信小程序登录
Aug 03 PHP
PHP与Perl之间知识点区别整理
Mar 19 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
关于mysql 字段的那个点为是定界符
2007/01/15 PHP
PHP读取MySQL数据代码
2008/06/05 PHP
php获得url参数中具有&amp;的值的方法
2014/03/05 PHP
百度地图API应用之获取用户的具体位置
2014/06/10 PHP
javascript 对表格的行和列都能加亮显示
2008/12/26 Javascript
默认让页面的第一个控件选中的javascript代码
2009/12/26 Javascript
jQuery 插件仿百度搜索框智能提示(带Value值)
2013/01/22 Javascript
原生js拖拽(第一课 未兼容)拖拽思路
2013/03/29 Javascript
关于eval 与new Function 到底该选哪个?
2013/04/17 Javascript
JQuery zClip插件实现复制页面内容到剪贴板
2015/11/02 Javascript
JavaScript实现自动切换图片代码
2016/10/11 Javascript
AngularJS ng-repeat数组有重复值的解决方法
2016/10/23 Javascript
Angular实现下拉框模糊查询功能示例
2018/01/03 Javascript
layui的布局和表格的渲染以及动态生成表格的方法
2019/09/18 Javascript
详解vue 组件的实现原理
2020/11/12 Javascript
详解Python3操作Mongodb简明易懂教程
2017/05/25 Python
pandas 获取季度,月度,年度首尾日期的方法
2018/04/11 Python
Python中利用xpath解析HTML的方法
2018/05/14 Python
Python爬虫实现使用beautifulSoup4爬取名言网功能案例
2019/09/15 Python
Python爬虫之Spider类用法简单介绍
2020/08/04 Python
10个python爬虫入门基础代码实例 + 1个简单的python爬虫完整实例
2020/12/16 Python
手把手教你用Django执行原生SQL的方法
2021/02/18 Python
css3中的calc函数浅析
2018/07/10 HTML / CSS
在HTML5中如何使用CSS建立不可选的文字
2014/10/17 HTML / CSS
New Balance澳大利亚官网:运动鞋和健身服装
2019/02/23 全球购物
Groupon西班牙官方网站:在线优惠券和交易,节省高达70%
2021/03/13 全球购物
人力资源管理毕业生自荐信
2013/11/21 职场文书
材料物理专业个人求职信
2013/12/15 职场文书
银行服务感言
2014/03/01 职场文书
库房管理员岗位职责
2014/03/09 职场文书
优秀共产党员先进事迹材料
2014/05/06 职场文书
学习型班组申报材料
2014/05/31 职场文书
订货会主持词
2015/07/01 职场文书
《认识钟表》教学反思
2016/02/16 职场文书
教你漂亮打印Pandas DataFrames和Series
2021/05/29 Python
vue+iview实现手机号分段输入框
2022/03/25 Vue.js