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中可能用来加密字符串的函数[base64_encode、urlencode、sha1]
Jan 16 PHP
提高php运行速度的一些小技巧分享
Jul 03 PHP
PHP在线生成二维码代码(google api)
Jun 03 PHP
php 生成自动创建文件夹并上传文件的示例代码
Mar 07 PHP
ThinkPHP的L方法使用简介
Jun 18 PHP
destoon官方标签大全
Jun 20 PHP
PHP PDOStatement对象bindpram()、bindvalue()和bindcolumn之间的区别
Nov 20 PHP
php+ajax实现无刷新数据分页的办法
Nov 02 PHP
PHP下SSL加密解密、验证、签名方法(很简单)
Jun 28 PHP
Yii框架表单提交验证功能分析
Jan 07 PHP
PHP中创建和编辑Excel表格的方法
Sep 13 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和ACCESS写聊天室(三)
2006/10/09 PHP
php curl post 时出现的问题解决
2014/01/30 PHP
php调用c接口无错版介绍
2014/03/11 PHP
浅析application/x-www-form-urlencoded和multipart/form-data的区别
2014/06/22 PHP
使用WordPress发送电子邮件的相关PHP函数用法解析
2015/12/15 PHP
php 一维数组的循环遍历实现代码
2017/04/10 PHP
php输出控制函数和输出函数生成静态页面
2019/06/27 PHP
PHP抽象类和接口用法实例详解
2019/07/20 PHP
PHP实现数组根据某个字段进行水平合并,横向合并案例分析
2019/10/08 PHP
JQuery入门——用bind方法绑定事件处理函数应用介绍
2013/02/05 Javascript
父页面显示遮罩层弹出半透明状态的dialog
2014/03/04 Javascript
jquery 为a标签绑定click事件示例代码
2014/06/23 Javascript
禁止按回车键提交表单的方法
2015/06/11 Javascript
JS实现仿新浪黄色经典滑动门效果代码
2015/09/27 Javascript
JS中判断某个字符串是否包含另一个字符串的五种方法
2018/05/03 Javascript
JavaScript面向对象的程序设计(犯迷糊的小羊)
2018/05/27 Javascript
原生js实现Flappy Bird小游戏
2018/12/24 Javascript
如何实现小程序tab栏下划线动画效果
2019/05/18 Javascript
微信小程序激励式视频广告组件使用详解
2019/12/06 Javascript
Vue 实现登录界面验证码功能
2020/01/03 Javascript
Vue axios 跨域请求无法带上cookie的解决
2020/09/08 Javascript
[01:10:16]DOTA2上海特级锦标赛B组资格赛#2 Fnatic VS Spirit第一局
2016/02/27 DOTA
python fabric使用笔记
2015/05/09 Python
Python提取支付宝和微信支付二维码的示例代码
2019/02/15 Python
Python搭建Spark分布式集群环境
2019/07/05 Python
python 实现查询Neo4j多节点的多层关系
2019/12/23 Python
Python3列表List入门知识附实例
2020/02/09 Python
python爬虫把url链接编码成gbk2312格式过程解析
2020/06/08 Python
HTML5的结构和语义(2):结构
2008/10/17 HTML / CSS
ONLY德国官方在线商店:购买时尚女装
2017/09/21 全球购物
洗车工岗位职责
2014/03/15 职场文书
公司门卫岗位职责范本
2014/07/08 职场文书
英语教师求职信范文
2015/03/20 职场文书
电影复兴之路观后感
2015/06/02 职场文书
导游词之京东大峡谷旅游区
2019/10/29 职场文书
pnpm对npm及yarn降维打击详解
2022/08/05 Javascript