当前位置: 首页 > 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.lqws.cn/news/546373.html

相关文章:

  • 小程序 API 开发手册:从入门到高级应用一网打尽
  • 基于[coze][dify]搭建一个智能体工作流,抓取热门视频数据,自动存入在线表格
  • Python打卡:Day38
  • 华为数通认证:适合谁的技术进阶之路?
  • 基于MySQL的分布式锁实现(Spring Boot + MyBatis)
  • 【数据分析,相关性分析】Matlab代码#数学建模#创新算法
  • 【C语言】知识总结·指针篇
  • 关于SAP产品名称变更通知 SAP云认证实施商工博科技
  • 动态控制click事件绑定
  • H.264中片数据分割(Slice Data Partitioning)介绍
  • Decoder-only PLM GPT1
  • c++异常
  • LINUX625 DNS反向解析
  • gemini-cli 踩坑实录
  • Windows VMWare Centos环境下安装Docker并配置MySql
  • PART 7 视频
  • web布局25
  • iOS打包流程中的安全处理实践:集成IPA混淆保护的自动化方案
  • 消息队列的网络模型详解:IO多路复用、Reactor模型、零拷贝
  • 一键获取服务器硬件脚本:CPU/内存/磁盘/RAID检测脚本详解
  • 电子行业 MES 系统:生产管理的智能引擎
  • Minio的扩容
  • 【docker】docker run参数说明
  • Imbalanced-learn 5. Ensemble of samplers
  • 【水印论文阅读1】将水印规则的定义域从离散的符号空间转移到连续的语义空间
  • 【大模型水印论文阅读2】前缀文本编码、均匀性约束
  • 【linux】程序地址空间
  • 信息抽取领域关键Benchmark方法:分类体系
  • 不同类型的微型导轨精度降低速度有何差异?
  • 专注搜索引擎优化的专业模板平台