当前位置: 首页 > news >正文

PHP「Not enough Memory」实战排错笔记

目录

PHP「Not enough Memory」实战排错笔记

1. 背景

2. 快速定位

3. 为什么 5 MB 的图片能耗尽 128 MB?

3.1 粗略估算公式(GD)

4. 实际峰值监控

5. 解决过程

6. 最佳实践与防御措施

7. 总结


PHP「Not enough Memory」实战排错笔记

——一次 5 MB 图片上传导致的内存溢出

1. 背景

  • 项目框架:Nginx + PHP-FPM 8.2

  • 文件管理器:Responsive Filemanager

  • 现象:上传一张 5 MB 的 JPEG 原图时,浏览器白屏,error_log 报:

    Not enough Memory (@/home/www/wwwroot/hnusri.cn/http/manager/plugins/ResponsiveFilemanager/filemanager/upload.php#241)
    
  • 默认配置memory_limit = 128M

2. 快速定位

  1. 开启详细日志

    display_errors = On
    log_errors     = On
    error_log      = /var/log/php/error.log
    
  2. 复现错误:上传同一张图片,观察 Peak memory(见 §4)。

3. 为什么 5 MB 的图片能耗尽 128 MB?

核心原因:GD 库在解码 / 缩放时会把整张图片展开到内存,按 4 byte/像素 计,再叠加中间缓冲。

3.1 粗略估算公式(GD)
memory ≈ 宽 × 高 × 4 × 1.65
  • 4:32 bit 色深

  • 1.65:经验系数,包含缩放 & 额外缓冲

分辨率文件体积*¹估算内存128 MB 足够吗
3840×2160 (4 K)≈5 MB3840×2160×4×1.65 ≈ 54 MB✔️
6000×4000 (24 MP)*²≈5 MB158 MB

*¹ JPEG 在磁盘上是压缩数据,跟解码内存无关。
*² 手机/单反随手拍常见 4–8 MB,但分辨率高达 20 ~ 30 MP。

4. 实际峰值监控

upload.php 适当位置插入:

register_shutdown_function(function () {error_log('Peak memory: ' . round(memory_get_peak_usage(true) / 1048576, 2) . ' MB');
});

再次上传,日志输出:

Peak memory: 163.14 MB

验证了公式推算。

5. 解决过程

  1. 调高 memory_limit

    memory_limit = 512M
    

    重启 PHP-FPM:

    sudo systemctl restart php-fpm
    

    再次上传,问题消失,峰值 163 MB 以内,留足裕量。

  2. 同步调整上传相关参数

    upload_max_filesize = 50M
    post_max_size       = 100M
    max_execution_time  = 300
    

6. 最佳实践与防御措施

措施说明建议级别
限制分辨率Responsive Filemanager 支持 $image_max_width / $image_max_height⭐⭐⭐⭐
使用 Imagickextension=imagick,解码时按实际色深,内存占用可降 40–60 %⭐⭐⭐
异步生成缩略图上传→消息队列→Worker 处理,避免前端线程内存峰值⭐⭐⭐
动态内存预算memory_limit ≈ 最大像素 × 4 × 1.65 × 并发系数⭐⭐
压缩上传前端或 App 先做分辨率压缩至 4 K 以内⭐⭐

7. 总结

  • 根因图片分辨率 决定解码峰值,而非磁盘体积。

  • 经验阈值:常见 24 MP 原图解码需 ~160 MB;并发 2 条就逼近 256 MB。

  • 最终 fix:将 memory_limit 提升至 512 MB 并优化上传策略,系统稳定运行至今。

如果你的线上环境仍保持默认 128 MB,而站点允许上传手机原图或单反照片,最好立即评估并调优内存策略。

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

相关文章:

  • 便利的隐形代价?智能设备正在“偷听”你的生活——物联网时代的隐私深度危机
  • window显示驱动开发—DirectX 图形内核子系统(三)
  • clion与keil分别配置项目宏定义
  • 问卷调查[mqtt dht]
  • 【C/C++】单元测试实战:Stub与Mock框架解析
  • AI+实时计算如何赋能金融系统?DolphinDB 在国泰君安期货年度中期策略会的演讲
  • 安心联车辆监控管理平台应用场景分析
  • IPv6配置
  • 每天一个前端小知识 Day 14 - 前端状态管理深入实践
  • 网络安全的两大威胁:XSS与CSRF攻击实例解析
  • 【大数据】大数据产品基础篇
  • 电脑远程控制另一台电脑无法连接怎么办
  • 【github】从本地更新仓库里的文件笔记
  • 微信小程序:实现树形结构组件
  • P27:RNN实现阿尔茨海默病诊断
  • Spring,Spring MVC,Spring Boot 之间什么关系?
  • Linux信号机制:从入门到精通
  • vscode把less文件生成css文件配置,设置生成自定义文件名称和路径
  • 移动端测试——如何解决iOS端无法打开弹窗式网页(Webkit)
  • 七、Python高级特性:迭代器、生成器与装饰器
  • 智能实验室革命:Deepoc大模型驱动全自动化科研新生态
  • 前端 E2E 测试实践:打造稳定 Web 应用的利器!
  • echarts柱状图要给柱子顶部加一个顶的写法方案
  • 在反向代理环境下精准获取客户端真实 IP 的最佳实践
  • 每日八股文补充2网络篇
  • GESP C++ 五级真题(2024年12月)题解
  • Shell 流程控制
  • 什么是Sanity Testing?和冒烟测试的区别?
  • Kotlin中协程挂起函数的本质
  • 数据结构学习——二叉树