当前位置:首页 > PHP教程 > php高级应用 > 列表

谈谈关于PHP内存溢出的思考

发布:smiling 来源: PHP粉丝网  添加日期:2022-06-06 08:29:32 浏览: 评论:0 

最近做大批量数据导出和数据导入的时候,经常会遇到PHP内存溢出的问题,在解决了问题之后,总结了一些经验,整理成文章记录下。

优化点

优化SQL语句,避免慢查询,合理的建立索引,查询指定的字段,sql优化这块在此就不展开了。

查询的结果集为大对象时转数组处理,框架中一般有方法可以转,如Laravel中有toArray(),Yii2中有asArray()。

对于大数组进行数据切割处理,PHP函数有array_chunk()、array_slice()。

对于大型的字符串和对象,使用引用传递&。

用过的变量及时unset。

导出的文件格式由excel改为csv

ini_set(‘memory_limit’,’’),设置程序可以使用的内存(不建议这样做)。

思考

内存管理

PHP的内存什么怎么管理的呢? 在学C语言时,开发者是需要手动管理内存。在PHP中,Zend引擎提供为了处理请求相关数据提供了一种特殊的内存管理器。请求相关数据是只需要服务单个请求,最迟会在请求结束时释放数据。

1.png

上图是来自于官网的描述截图

防止内存泄漏并尽可能快地释放所有内存是内存管理的重要组成部分。因为安全原因,Zend引擎会释放所有上面提到的API锁分配的内存。

垃圾回收机制

简单说下:

PHP5.3之前,采用引用计数的方式管理。PHP中的变量存在zval的变量容器中,变量被引用的时,引用计数+1,变量引用计数为0时,PHP将在内存中销毁这个变量。但是在引用计数循环引用时,引用计数就不会消减为0,导致内存泄漏。

PHP5.3之后做了优化,并不是每次引用计数减少都进入回收周期,只有根缓冲区满额后才开始进行垃圾回收,这样可以解决循环引用的问题,也可以将总内存泄漏保持在一个阈值之下。

代码

由于使用phpexcel时经常会遇到内存溢出,下面分享一段生成csv文件的代码:

  1. <?php 
  2.  
  3.  
  4.  
  5. namespace api\service; 
  6.  
  7.  
  8.  
  9. class ExportService 
  10.  
  11.  
  12.  
  13.  
  14.     public static $outPutFile = ''
  15.  
  16.  
  17.  
  18.     /** 
  19.  
  20.      * 导出文件 
  21.  
  22.      * @param string $fileName 
  23.  
  24.      * @param $data 
  25.  
  26.      * @param array $formFields 
  27.  
  28.      * @return mixed 
  29.  
  30.      */ 
  31.  
  32.     public static function exportData($fileName = ''$data$formFields = []) 
  33.  
  34.     { 
  35.  
  36.         $fileArr = []; 
  37.  
  38.         $tmpPath = \Yii::$app->params['excelSavePath']; 
  39.  
  40.  
  41.  
  42.         foreach (array_chunk($data, 10000) as $key => $value) { 
  43.  
  44.             self::$outPutFile = ''
  45.  
  46.             $subject          = !emptyempty($fileName) ? $fileName : 'data_'
  47.  
  48.             $subject          .= date('YmdHis'); 
  49.  
  50.             if (emptyempty($value) || emptyempty($formFields)) { 
  51.  
  52.                 continue
  53.  
  54.             } 
  55.  
  56.  
  57.  
  58.             self::$outPutFile = $tmpPath . $subject . $key . '.csv'
  59.  
  60.             if (!file_exists(self::$outPutFile)) { 
  61.  
  62.                 touch(self::$outPutFile); 
  63.  
  64.             } 
  65.  
  66.             $index  = array_keys($formFields); 
  67.  
  68.             $header = array_values($formFields); 
  69.  
  70.             self::outPut($header); 
  71.  
  72.  
  73.  
  74.             foreach ($value as $k => $v) { 
  75.  
  76.                 $tmpData = []; 
  77.  
  78.                 foreach ($index as $item) { 
  79.  
  80.                     $tmpData[] = isset($v[$item]) ? $v[$item] : ''
  81.  
  82.                 } 
  83.  
  84.                 self::outPut($tmpData); 
  85.  
  86.             } 
  87.  
  88.             $fileArr[] = self::$outPutFile
  89.  
  90.         } 
  91.  
  92.           
  93.  
  94.         $zipFile = $tmpPath . $fileName . date('YmdHi') . '.zip'
  95.  
  96.         $zipRes = self::zipFile($fileArr$zipFile); 
  97.  
  98.         return $zipRes
  99.  
  100.     } 
  101.  
  102.  
  103.  
  104.     /** 
  105.  
  106.      * 向文件写入数据 
  107.  
  108.      * @param array $data 
  109.  
  110.      */ 
  111.  
  112.     public static function outPut($data = []) 
  113.  
  114.     { 
  115.  
  116.         if (is_array($data) && !emptyempty($data)) { 
  117.  
  118.             $data = implode(','$data); 
  119.  
  120.             file_put_contents(self::$outPutFile, iconv("UTF-8""GB2312//IGNORE"$data) . PHP_EOL, FILE_APPEND); 
  121.  
  122.         } 
  123.  
  124.     } 
  125.  
  126.  
  127.  
  128.     /** 
  129.  
  130.      * 压缩文件 
  131.  
  132.      * @param $sourceFile 
  133.  
  134.      * @param $distFile 
  135.  
  136.      * @return mixed 
  137.  
  138.      */ 
  139.  
  140.     public static function zipFile($sourceFile$distFile
  141.  
  142.     { 
  143.  
  144.         $zip = new \ZipArchive(); 
  145.  
  146.         if ($zip->open($distFile, \ZipArchive::CREATE) !== true) { 
  147.  
  148.             return $sourceFile
  149.  
  150.         } 
  151.  
  152.  
  153.  
  154.         $zip->open($distFile, \ZipArchive::CREATE); 
  155.  
  156.         foreach ($sourceFile as $file) { 
  157.  
  158.             $fileContent = file_get_contents($file); 
  159.  
  160.             $file        = iconv('utf-8''GBK'basename($file)); 
  161.  
  162.             $zip->addFromString($file$fileContent); 
  163.  
  164.         } 
  165.  
  166.         $zip->close(); 
  167.  
  168.         return $distFile
  169.  
  170.     } 
  171.  
  172.       
  173.  
  174.         /** 
  175.  
  176.      * 下载文件 
  177.  
  178.      * @param $filePath 
  179.  
  180.      * @param $fileName 
  181.  
  182.      */ 
  183.  
  184.     public static function download($filePath$fileName
  185.  
  186.     { 
  187.  
  188.         if (!file_exists($filePath . $fileName)) { 
  189.  
  190.             header('HTTP/1.1 404 NOT FOUND'); 
  191.  
  192.         } else { 
  193.  
  194.             //以只读和二进制模式打开文件 
  195.  
  196.             $file = fopen($filePath . $fileName"rb"); 
  197.  
  198.  
  199.  
  200.             //告诉浏览器这是一个文件流格式的文件 
  201.  
  202.             Header("Content-type: application/octet-stream"); 
  203.  
  204.             //请求范围的度量单位 
  205.  
  206.             Header("Accept-Ranges: bytes"); 
  207.  
  208.             //Content-Length是指定包含于请求或响应中数据的字节长度 
  209.  
  210.             Header("Accept-Length: " . filesize($filePath . $fileName)); 
  211.  
  212.             //用来告诉浏览器,文件是可以当做附件被下载,下载后的文件名称为$file_name该变量的值 
  213.  
  214.             Header("Content-Disposition: attachment; filename=" . $fileName); 
  215.  
  216.  
  217.  
  218.             //读取文件内容并直接输出到浏览器 
  219.  
  220.             echo fread($filefilesize($filePath . $fileName)); 
  221.  
  222.             fclose($file); 
  223.  
  224.             exit(); 
  225.  
  226.         } 
  227.  
  228.     } 
  229.  

调用出代码

  1. $fileName = "库存导入模板"
  2.  
  3. $stockRes = []; // 导出的数据 
  4.  
  5. $formFields = [ 
  6.  
  7.     'store_id'  => '门店ID'
  8.  
  9.     'storeName' => '门店名称'
  10.  
  11.     'sku'       => 'SKU编码'
  12.  
  13.     'name'      => 'SKU名称'
  14.  
  15.     'stock'     => '库存'
  16.  
  17.     'reason'    => '原因' 
  18.  
  19. ]; 
  20.  
  21. $fileRes    = ExportService::exportData($fileName$stockRes$formFields); 
  22.  
  23. $tmpPath    = \Yii::$app->params['excelSavePath']; // 文件路径 
  24.  
  25. $fileName   = str_replace($tmpPath''$fileRes); 
  26.  
  27.  
  28.  
  29. // 下载文件 
  30.  
  31. ExportService::download($tmpPath$fileName);

Tags: PHP内存溢出

分享到: