page contents

记因PHP的内存溢出导致的事故之解决

今天上午刚到公司,就有同事在公司群里反映某个计划任务出现问题了。我就怀着刨根问底的心,去查看了log。发现挺有意思的一个问题,PHP内存溢出导致脚本执行失败。那就一起来看个究竟吧!

attachments-2021-02-IZEpfiXa601a17f3c41c4.png

今天上午刚到公司,就有同事在公司群里反映某个计划任务出现问题了。我就怀着刨根问底的心,去查看了log。发现挺有意思的一个问题,PHP内存溢出导致脚本执行失败。那就一起来看个究竟吧!


1. 首先查看了计划任务的Log

image

从报错信息字面意思可以看出,允许的134217728 bytes的内存已经用尽,还要试图分配12961640 bytes
内存。

给你(当前脚本)分配的内存你已经用完了,你还想问系统要内存。系统这时想对你说:

地主家也没有余粮啊(借用葛优大爷的一句话)

image

1. 模拟一下"案发现场"

  • 新建一个mem_exhausted.php文件 copy过来一个2.4M的log文件做测试用
  • 写个简单的脚本重现"案发现场" 故意分配1M的内存 来读取2.4M的log
  • 执行脚本,"案发现场"重现

image

image

image

1. 分析"事故"原因

脚本一次性读取了大量的数据(可能是读的文件,可能是读取的数据库)

如下图: 往杯子(分配给当前脚本的内存)里面倒数水(log文件的数据),杯子容量(内存)不够用

2. 解决方案

a. 既然杯子小 就换个大杯子(增大给脚本分配的内存)治标不治本: `ini_set('memory_limit','100M');`

image

image

b. 把水分批次倒入杯子中(循环,分段读取数据,读数据库的话可以用limit)

image

看看结果

image

分段读取也是可以解决问题滴

其他优化方案

  • 应当尽可能减少静态变量的使用,在需要数据重用时,可以考虑使用引用(&)。
  • 数据库操作完成后,要马上关闭连接;
  • 一个对象使用完,要及时调用析构函数(__destruct())
  • 用过的变量及时销毁(unset())掉
  • 可以使用memory_get_usage()函数,获取当前占用内存 根据当前使用的内存来调整程序
  • unset()函数只能在变量值占用内存空间超过256字节时才会释放内存空间。(PHP内核的gc垃圾回收机制决定)
  • 有当指向该变量的所有变量(如引用变量)都被销毁后,才会释放内存

(PHP变量底层实现是一个_zval_struct结构体,is_ref__gc表示引用计数 is_ref__gc表示是否为引用)


attachments-2021-02-i8lBZLQp601a1814993be.jpg来源:https://www.cnblogs.com/gaohj/p/6727069.html

  • 发表于 2021-02-03 11:26
  • 阅读 ( 604 )
  • 分类:PHP开发

0 条评论

请先 登录 后评论
轩辕小不懂
轩辕小不懂

2403 篇文章

作家榜 »

  1. 轩辕小不懂 2403 文章
  2. 小柒 1320 文章
  3. Pack 1135 文章
  4. Nen 576 文章
  5. 王昭君 209 文章
  6. 文双 71 文章
  7. 小威 64 文章
  8. Cara 36 文章