项目场景:在原有的基础上追加excel数据
描述:分批导进数据,从而减少内存的消耗,将新增的数据填充进对应单元格中。问题描述:
提示:导出数据还是会出现内存溢出
原因分析:
文件写入都离不开:1.打开文件->2.写入文件的流程
在打开文件流程中,需要将所有数据加载一遍,当文件越大,就会出现内存溢出。所以追加数据操作还是限制于打开文件的过程。
可以通过register_shutdown_function注册回调函数可以捕捉到内存溢出的问题,我写了个小案例。这样子就可以在使用的框架中捕捉到内存溢出导致的问题。
问题1:为什么还没有内存爆满,但却已经抛出异常?
系统设置了6M的内存,但是我执行这些代码使用memory_get_usage查询得到系统只是消耗了不到5M的内存。是不是可以看成php自动释放了一些内存,或者是有个预警值,再或者注册某些东西需要占位内存。这个先不讨论,以后找到再补充。我尝试了几个版本,消耗的内存都没有百分百消耗。
解决方案:基于register_shutdown_function能捕捉异常,并且执行回调逻辑的功能。
我可以在我的代码中捕捉内存溢出的异常。
注意:由于问题1的存在,所以回调中不能执行太复杂的逻辑,太复杂也会内存溢出,但可以动态加点内存再来执行的逻辑
多次注册register_shutdown_function,他会按顺序执行注册的回调函数方法。thinkphp的回调中,会出现较复杂的逻辑,这时就会出现执行的代码又出现内存溢出,日志写不进,报错报不显示的问题,那么我们可以在执行代码的地方先注册一个register_shutdown_function,例如tp6的cli方法,可以在config文件下的console.php文件捕捉内存溢出操作,从而避免回调函数时内存不足导致的日志写不进去。
先执行console.php的register_shutdown_function,再执行框架的register_shutdown_function。