PHP的垃圾回收机制代码实例讲解


Posted in PHP onFebruary 27, 2021

PHP可以自动进行内存管理,清除不需要的对象,主要使用了引用计数

zval结构体中定义了ref_countis_ref , ref_count是引用计数 ,标识此zval被多少个变量引用 , 为0时会被销毁
is_ref标识是否使用的 &取地址符强制引用

为了解决循环引用内存泄露问题 , 使用同步周期回收算法
比如当数组或对象循环的引用自身 , unset掉数组的时候 , 当refcount-1后还大于0的 , 就会被当成疑似垃圾 , 会进行遍历 ,并且模拟的删除一次refcount-1如果是0就删除 ,如果不是0就恢复

顽固垃圾的产生过程

<?php
  $a = "new string";
?>

代码中,$a变量内部存储信息为

a: (refcount_gc=1, is_ref_gc=0)='new string'

当把 a 赋 值 给 另 外 一 个 变 量 的 时 候 , a赋值给另外一个变量的时候, a赋值给另外一个变量的时候,a对应的zval的refcount_gc会加1

<?php
  $a = "new string";
  $b = $a;
?>

此时 a 和 a和 a和b变量对应的内部存储信息为, a 和 a和 a和b同时指向一个字符串"new string" ,它的refcount变成2

a,b: (refcount_gc=2, is_ref=0)='new string'

当用unset删除$b变量时,“new string” 的refcount_gc会减1变成1。

<?php
  $a = "new string"; //a: (refcount_gc=1, is_ref_gc=0)='new string'
  $b = $a;      //a,b: (refcount_gc=2, is_ref=0)='new string'
  unset($b);     //a: (refcount_gc=1, is_ref=0)='new string'
?>

对于普通的变量来说,这一切很正常,但是在复合类型变量(数组和对象)中,会发生比较有意思的事情:

<?php
  $a = array('meaning' => 'life', 'number' => 42);
?>

$a内部存储信息为:

a: (refcount=1, is_ref=0)=array (
'meaning' => (refcount=1, is_ref=0)='life',
'number' => (refcount=1, is_ref=0)=42
)

数组变量本身($a)在引擎内部实际上是一个哈希表,这张表中有两个zval项 meaning和number,所以实际上那一行代码中一共生成了3个zval,这3个zval都遵循变量的引用和计数原则,用图来表示:
PHP的垃圾回收机制代码实例讲解

下面在$a中添加一个元素,并将现有的一个元素的值赋给新的元素:

<?php
  $a = array('meaning' => 'life', 'number' => 42);
  $a['name'] = $a['meaning'];
?>

那么$a的内部存储为 , “life” 的ref_count变成2 , 42的ref_count是1:

a: (refcount=1, is_ref=0)=array (
'meaning' => (refcount=2, is_ref=0)='life',
'number' => (refcount=1, is_ref=0)=42,
'name' => (refcount=2, is_ref=0)='life'
)

如果将数组的引用赋值给数组中的一个元素,有意思的事情就会发生:

<?php
  $a = array('one');
  $a[] = &$a;
?>

这样 a 数 组 就 有 两 个 元 素 , 一 个 索 引 为 0 , 值 为 字 符 o n e , 另 外 一 个 索 引 为 1 , 为 a数组就有两个元素,一个索引为0,值为字符one,另外一个索引为1,为 a数组就有两个元素,一个索引为0,值为字符one,另外一个索引为1,为a自身的引用,内部存储如下:
PHP的垃圾回收机制代码实例讲解

a: (refcount=2, is_ref=1)=array (
0 => (refcount=1, is_ref=0)='one',
1 => (refcount=2, is_ref=1)=…
)

array这个zvalref_count是2 , 是一个环形引用
这时对$a进行unset,那么 a 会 从 符 号 表 中 删 除 , 同 时 ‘ a会从符号表中删除,同时` a会从符号表中删除,同时‘a指向的zvalrefcount_gc`减少1.

<?php
$a = array('one');
$a[] = &$a;
unset($a);
?>

那么问题就产生了, a 已 经 不 在 符 号 表 中 , 用 户 无 法 再 访 问 此 变 量 , 但 是 a已经不在符号表中,用户无法再访问此变量,但是 a已经不在符号表中,用户无法再访问此变量,但是a之前指向的zval的refcount_gc变为1而不是0,因此不能被回收,从而产生内存泄露,新的GC要做的工作就是清理此类垃圾。

为了解决循环引用内存泄露问题 , 使用同步周期回收算法 , 这种ref_count减1后还大于0的会被作为疑似垃圾

比如当数组或对象循环的引用自身 , unset掉数组的时候 , 当refcount-1后还大于0的 , 会进行遍历 ,并且模拟的删除一次refcount-1如果是0就删除 ,如果不是0就恢复。

到此这篇关于PHP的垃圾回收机制代码实例讲解的文章就介绍到这了,更多相关PHP的垃圾回收机制内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

PHP 相关文章推荐
PHP执行批量mysql语句的解决方法
May 02 PHP
php实现查询百度google收录情况(示例代码)
Aug 02 PHP
php打开远程文件的方法和风险及解决方法
Nov 12 PHP
php判断电脑访问、手机访问的例子
May 10 PHP
php基于curl扩展制作跨平台的restfule 接口
May 11 PHP
PHP脚本监控Nginx 502错误并自动重启php-fpm
May 13 PHP
php实现网站文件批量压缩下载功能
Oct 28 PHP
PHP邮件群发机实现代码
Feb 16 PHP
PHP页面跳转操作实例分析(header方法)
Sep 28 PHP
magento后台无法登录解决办法的两种方法
Dec 09 PHP
thinkphp5+layui实现的分页样式示例
Oct 08 PHP
laravel 解决Validator使用中出现的问题
Oct 25 PHP
php命令行模式代码实例详解
Feb 26 #PHP
PHP时间类完整代码实例
Feb 26 #PHP
PHP队列场景以及实现代码实例详解
Feb 26 #PHP
PHP实现长轮询消息实时推送功能代码实例讲解
Feb 26 #PHP
php的对象传值与引用传值代码实例讲解
Feb 26 #PHP
php并发加锁问题分析与设计代码实例讲解
Feb 26 #PHP
PHP内存溢出优化代码详解
Feb 26 #PHP
You might like
通过5个php实例细致说明传值与传引用的区别
2012/08/08 PHP
PHP+Mysql+Ajax实现淘宝客服或阿里旺旺聊天功能(前台页面)
2017/06/16 PHP
PHP isset empty函数相关面试题及解析
2020/12/11 PHP
javascript 四则运算精度修正函数代码
2010/05/31 Javascript
11款新鲜的jQuery插件[附所有demo下载]
2011/01/24 Javascript
jquery $.each()使用探讨
2013/09/23 Javascript
jquery获取一组checkbox的值(实例代码)
2013/11/04 Javascript
PHP abstract与interface之间的区别
2013/11/11 Javascript
兼容主流浏览器的iframe自适应高度js脚本
2014/01/10 Javascript
javascript trim函数在IE下不能用的解决方法
2014/09/12 Javascript
基于JavaScript如何制作遮罩层对话框
2016/01/26 Javascript
Bootstrap4一次重大更新 几乎涉及每行代码
2016/05/16 Javascript
angularJS 如何读写缓冲的方法(推荐)
2016/08/06 Javascript
详解Angularjs在控制器(controller.js)中使用过滤器($filter)格式化日期/时间实例
2017/02/17 Javascript
js实现下拉菜单效果
2017/03/01 Javascript
茶余饭后聊聊Vue3.0响应式数据那些事儿
2019/10/30 Javascript
Vue组件化开发之通用型弹出框的实现
2020/02/28 Javascript
在vue中使用Echarts画曲线图的示例
2020/10/03 Javascript
Linux下python与C++使用dlib实现人脸检测
2018/06/29 Python
python使用MQTT给硬件传输图片的实现方法
2019/05/05 Python
Python计算一个点到所有点的欧式距离实现方法
2019/07/04 Python
如何使用Python自动控制windows桌面
2019/07/11 Python
python的flask框架难学吗
2020/07/31 Python
意大利奢侈品网站:Italist
2016/08/23 全球购物
美国便宜的横幅和标志印刷在线:Best of Signs
2019/05/29 全球购物
在DELPHI中调用存储过程和使用内嵌SQL哪种方式更好
2016/11/22 面试题
优秀学生干部个人的自我评价
2013/10/04 职场文书
汽车专业毕业生自荐信
2013/11/03 职场文书
医学生自我鉴定范文
2013/11/08 职场文书
安全标准化实施方案
2014/02/20 职场文书
互联网电子商务专业毕业生求职信
2014/03/18 职场文书
《金色的脚印》教后反思
2014/04/23 职场文书
人事专员岗位说明书
2014/07/29 职场文书
JS ES6异步解决方案
2021/04/29 Javascript
详细介绍python类及类的用法
2021/05/31 Python
Win10服务全部禁用了怎么启动?Win10服务全部禁用解决方法
2022/09/23 数码科技