PHP 垃圾回收高级特性

PHP 垃圾回收高级特性

1. 循环引用与内存泄漏

单纯的引用计数在遇到循环引用时会导致内存泄漏,主要原因是引用计数无法正确识别那些仅通过循环引用相互关联但实际上已经不可达的对象。

1.1 引用计数的基本原理

引用计数是一种内存管理机制,通过维护每个对象的引用计数来决定对象是否可以被销毁:

  • 创建对象:引用计数初始为 1
  • 新增引用:引用计数增加
  • 删除引用:引用计数减少
  • 销毁对象:当引用计数变为 0 时,对象被销毁,其内存被回收

2. 循环引用详解

2.1 循环引用的定义

循环引用是指两个或多个对象相互引用,形成一个闭环。例如:

$a = new UsersEntity();
$b = new UsersEntity();
$a->ref = $b; // $a 引用了 $b
$b->ref = $a; // $b 引用了 $a

在这个例子中:

  • $a 的引用计数为 1(被 $b->ref 引用)
  • $b 的引用计数为 1(被 $a->ref 引用)

注意:即使脚本中不再使用 $a 和 $b,它们的引用计数都不会变为 0,因为它们相互引用。

2.2 引用计数的局限性

引用计数无法判断循环引用对象是否真正被程序所需。即使这些对象在逻辑上不可达(没有外部引用指向它们),它们之间的引用关系仍然会导致引用计数始终大于 0。


3. 内存泄漏示例

3.1 基本示例

gc_enable(); // 启用垃圾回收
$a = new UsersEntity();
$b = new UsersEntity();
$a->ref = $b; // 循环引用
$b->ref = $a;unset($a);
unset($b);

重要:即使没有手动触发垃圾回收,这里也会出现内存泄漏。即使 $a 和 $b 已经被 unset,它们仍然在相互引用,引用计数器无法减少到 0。

3.2 实际应用示例

public function getStatusWithCycle()
{gc_enable(); // 启用垃圾回收$a = new UsersEntity();$b = new UsersEntity();$a->ref = $b; // 循环引用$b->ref = $a;unset($a);unset($b);$endStatus = gc_status();return ['end_status' => $endStatus];
}

返回结果示例:

{"data": {"end_status": {"runs": 0,"collected": 0,"threshold": 10001,"roots": 2433}}
}

4. 垃圾回收器的解决方案

为了弥补引用计数的局限性,PHP 引入了垃圾回收器(GC),采用了基于根集合(roots)和可达性分析的算法:

  1. 根集合:程序中所有可以直接访问的对象
  2. 标记阶段:遍历根集合,标记所有可达的对象
  3. 清除阶段:回收未被标记的对象,包括循环引用的对象
gc_collect_cycles(); // 手动触发垃圾回收

5. 自动垃圾回收机制

如果启用了垃圾回收机制,即使没有手动调用 gc_collect_cycles(),理论上内存溢出的风险大大降低,但仍然可能发生,取决于以下因素:

5.1 自动垃圾回收触发条件

  • PHP 的垃圾回收器在运行时会自动检测是否需要回收循环引用的内存资源
  • 垃圾回收的触发基于根集合的增长(roots)和预定义的阈值(gc_status()['threshold']
  • 如果 roots 增长未达到 threshold,垃圾回收不会触发

注意:如果代码中循环引用对象的生成速度超过垃圾回收器的触发速度,可能出现短期内的内存占用高峰甚至溢出。

5.2 脚本运行时长和负载

短生命周期脚本
  • 大多数 PHP 网页脚本属于此类
  • 脚本结束时会清理所有内存,包括循环引用的对象
  • 通常不会内存溢出,但可能出现瞬间内存使用过高
长生命周期脚本
  • 守护进程、队列处理器、WebSocket 服务等
  • 可能持续运行并产生大量循环引用对象
  • 如果垃圾回收未及时触发,内存使用会逐渐增加

6. 内存管理最佳实践

6.1 如何降低内存溢出风险

  1. 配置优化

    • 确保 memory_limit 配置足够高
    • 适当调整垃圾回收阈值
  2. 代码优化

    • 避免频繁创建循环引用对象
    • 及时打破不必要的引用关系
  3. 主动管理

    • 使用 unset() 及时打破引用关系
    • 在适当位置手动触发垃圾回收

6.2 监控和优化建议

  1. 内存监控

    • 定期检查 gc_status() 的 roots 和 collected 值
    • 引入内存监控和日志机制
  2. 性能优化

    • 优化代码结构
    • 减少不必要的对象创建
    • 适当调用 gc_collect_cycles()

总结

关键要点

  • 循环引用是引用计数机制的主要缺陷
  • PHP 的垃圾回收器通过可达性分析解决循环引用问题
  • 合理使用手动垃圾回收和内存监控可有效预防内存溢出
  • 在高负载场景下需要特别注意内存管理

通过优化代码结构和适当调用 gc_collect_cycles(),可以有效避免内存溢出问题。在实际应用中,应结合具体场景选择合适的内存管理策略。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/bicheng/83002.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

奈雪小程序任务脚本

功能概述 该脚本用于自动完成奈雪点单小程序的每日任务,包括: 自动检测 Token 有效性自动签到(如果未签到)获取用户基础信息(昵称、手机号)查询当前奈雪币余额记录连续签到天数支持多账号执行&#xff0c…

基于cornerstone3D的dicom影像浏览器 第二十七章 设置vr相机,复位视图

文章目录 前言一、VR视图设置相机位置1. 相机位置参数2. 修改mprvr.js3. 调用流程1) 修改Toolbar3D.vue2) 修改View3d.vue3) 修改DisplayerArea3D.vue 二、所有视图复位1.复位流程说明2. 调用流程1) Toolbar3D中添加"复位"按钮,发送reset事件2) View3d.vu…

Opencv4 c++ 自用笔记 03 滑动条、相机与视频操作

1. 相机与视频操作 1.1 打开视频/相机 OpenCV 中 imread() 只能读取静态图像,若要读取视频文件或摄像头流,需要使用 VideoCapture 类: // 构造函数 cv::VideoCapture::VideoCapture(); cv::VideoCapture…

身份证发给别人怎么加水印?赛文奥特曼身份证添加水印教程

我们经常需要使用身份证照片进行身份验证、资料提交等操作。然而,直接将身份证照片发送给他人或上传到网络存在一定的信息泄露风险。为了更好地保护个人隐私,我们可以使用 简鹿水印助手 这款工具,在身份证照片上添加专属水印,从而…

十、【核心功能篇】项目与模块管理:前端页面开发与后端 API 联调实战

【核心功能篇】项目与模块管理:前端页面开发与后端 API 联调实战 前言准备工作第一部分:完善项目管理功能 (Project)1. 创建/编辑项目的表单对话框组件 第二部分:模块管理功能 (集成到项目详情页)1. 创建模块相关的 API 服务 (src/api/module…

ES分词搜索

ES的使用 前言作者使用的版本作者需求 简介ES简略介绍ik分词器简介 使用es的直接简单使用es的查询 es在java中使用备注说明 前言 作者使用的版本 es: 7.17.27spring-boot-starter-data-elasticsearch: 7.14.2 作者需求 作者接到一个业务需求,我们系统有份数据被…

Axure设计案例——科技感立体柱状图

想让你的数据展示告别平淡无奇,成为吸引全场目光的焦点吗?快来瞧瞧这个Axure设计的科技感立体柱状图案例!科技感设计风格借助逼真的立体效果打破传统柱状图的平面感,营造出一种令人眼前一亮的视觉震撼。每一个柱状体都仿佛是真实存…

恶意npm与VS Code包窃取数据及加密货币资产

60个npm包窃取系统敏感信息 安全研究人员在npm软件包注册表中发现60个恶意组件,这些组件能够收集主机名、IP地址、DNS服务器和用户目录信息,并将其发送至Discord平台控制的终端节点。据Socket安全研究员Kirill Boychenko上周发布的报告显示,…

leetcode 2359. 找到离给定两个节点最近的节点

给你一个 n 个节点的 有向图 ,节点编号为 0 到 n - 1 ,每个节点 至多 有一条出边。 有向图用大小为 n 下标从 0 开始的数组 edges 表示,表示节点 i 有一条有向边指向 edges[i] 。如果节点 i 没有出边,那么 edges[i] -1 。 同时…

1. pytorch手写数字预测

1. pytorch手写数字预测 1.背景2.准备数据集2.定义模型3.dataloader和训练4.训练模型5.测试模型6.保存模型 1.背景 因为自身的研究方向是多模态目标跟踪,突然对其他的视觉方向产生了兴趣,所以心血来潮的回到最经典的视觉任务手写数字预测上来&#xff0…

AWS WebRTC:获取ICE服务地址(part 2): ICE Agent的作用

上一篇,已经获取到了ICE服务地址,从返回结果中看,是两组TURN服务地址。 拿到这些地址有什么用呢?接下来就要说到WebRTC中ICE Agent的作用了,返回的服务地址会传给WebRTC最终给到ICE Agent。 ICE Agent的作用&#xf…

大数据时代的利剑:Bright Data网页抓取与自动化工具共建高效数据采集新生态

目录 一、为何要选用Bright Data网页自动化抓取——帮助我们高效高质解决以下问题! 二、Bright Data网页抓取工具 - 网页爬虫工具实测 2.1 首先注册用户 2.2 首先点击 Proxies & Scraping ,再点击浏览器API的开始使用 2.3 填写通道名称&#xff…

指纹识别+精准化POC攻击

开发目的 解决漏洞扫描器的痛点 第一就是扫描量太大,对一个站点扫描了大量的无用 POC,浪费时间 指纹识别后还需要根据对应的指纹去进行 payload 扫描,非常的麻烦 开发思路 我们的思路分为大体分为指纹POC扫描 所以思路大概从这几个方面…

【Day40】

DAY 40 训练和测试的规范写法 知识点回顾: 彩色和灰度图片测试和训练的规范写法:封装在函数中展平操作:除第一个维度batchsize外全部展平dropout操作:训练阶段随机丢弃神经元,测试阶段eval模式关闭dropout 作业&#x…

【HTML-13】HTML表格合并技术详解:打造专业数据展示

表格是HTML中展示结构化数据的重要元素,而表格合并则是提升表格表现力的关键技术。本文将全面介绍HTML中的表格合并方法,帮助您创建更专业、更灵活的数据展示界面。 1. 表格合并基础概念 在HTML中,表格合并主要通过两个属性实现&#xff1a…

<uniapp><threejs>在uniapp中,怎么使用threejs来显示3D图形?

前言 本专栏是基于uniapp实现手机端各种小功能的程序,并且基于各种通讯协议如http、websocekt等,实现手机端作为客户端(或者是手持机、PDA等),与服务端进行数据通讯的实例开发。 发文平台 CSDN 环境配置 系统:windows 平台:visual studio code、HBuilderX(uniapp开…

如何制作全景VR图?

全景VR图,特别是720度全景VR,为观众提供一种沉浸式体验。 全景VR图能够捕捉场景的全貌,还能将多个角度的图片或视频无缝拼接成一个完整的全景视角,让观众在虚拟环境中自由探索。随着虚拟现实(VR)技术的飞速…

前端使用qrcode来生成二维码的时候中间添加logo图标

这个开源仓库可以让你在前端页面中生成二维码图片,并且支持调整前景色和背景色,但是有个问题,就是不能添加logo图片。issue: GitHub Where software is built 但是已经有解决方案了: add a logo picture Issue #21…

【C语言】函数指针及其应用

目录 1.1 函数指针的概念和应用 1.2 赋值与内存模型 1.3 调用方式与注意事项 二、函数指针的使用 2.1 函数指针的定义和访问 2.2 动态调度:用户输入驱动函数执行 2.3 函数指针数组进阶应用 2.4 函数作为参数的高阶抽象 三、回调函数 3.1 指针函数…

安装flash-attention失败的终极解决方案(WINDOWS环境)

想要看linux版本下安装问题的请走这里:安装flash-attention失败的终极解决方案(LINUX环境) 其实,现在的flash-attention不像 v2.3.2之前的版本,基本上不兼容WINDOWS环境。但是在WINDOWS环境安装总还是有那么一点不顺畅…