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

PHP内存溢出优化代码详解

发布:smiling 来源: PHP粉丝网  添加日期:2022-04-13 13:34:15 浏览: 评论:0 

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

优化点

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

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

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

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

5、用过的变量及时unset。

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

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

内存管理

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

PHP内存溢出优化代码详解

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

垃圾回收机制

简单说下:

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

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

代码

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

  1. <?php 
  2.  
  3. namespace api\service; 
  4.  
  5. class ExportService 
  6.  
  7.  public static $outPutFile = ''
  8.  
  9.  /** 
  10.  * 导出文件 
  11.  * @param string $fileName 
  12.  * @param $data 
  13.  * @param array $formFields 
  14.  * @return mixed 
  15.  */ 
  16.  public static function exportData($fileName = ''$data$formFields = []) 
  17.  { 
  18.  $fileArr = []; 
  19.  $tmpPath = \Yii::$app->params['excelSavePath']; 
  20.  
  21.  foreach (array_chunk($data, 10000) as $key => $value) { 
  22.   self::$outPutFile = ''
  23.   $subject  = !emptyempty($fileName) ? $fileName : 'data_'
  24.   $subject  .= date('YmdHis'); 
  25.   if (emptyempty($value) || emptyempty($formFields)) { 
  26.   continue
  27.   } 
  28.  
  29.   self::$outPutFile = $tmpPath . $subject . $key . '.csv'
  30.   if (!file_exists(self::$outPutFile)) { 
  31.   touch(self::$outPutFile); 
  32.   } 
  33.   $index = array_keys($formFields); 
  34.   $header = array_values($formFields); 
  35.   self::outPut($header); 
  36.  
  37.   foreach ($value as $k => $v) { 
  38.   $tmpData = []; 
  39.   foreach ($index as $item) { 
  40.    $tmpData[] = isset($v[$item]) ? $v[$item] : ''
  41.   } 
  42.   self::outPut($tmpData); 
  43.   } 
  44.   $fileArr[] = self::$outPutFile
  45.  } 
  46.    
  47.  $zipFile = $tmpPath . $fileName . date('YmdHi') . '.zip'
  48.  $zipRes = self::zipFile($fileArr$zipFile); 
  49.  return $zipRes
  50.  } 
  51.  
  52.  /** 
  53.  * 向文件写入数据 
  54.  * @param array $data 
  55.  */ 
  56.  public static function outPut($data = []) 
  57.  { 
  58.  if (is_array($data) && !emptyempty($data)) { 
  59.   $data = implode(','$data); 
  60.   file_put_contents(self::$outPutFile, iconv("UTF-8""GB2312//IGNORE"$data) . PHP_EOL, FILE_APPEND); 
  61.  } 
  62.  } 
  63.  
  64.  /** 
  65.  * 压缩文件 
  66.  * @param $sourceFile 
  67.  * @param $distFile 
  68.  * @return mixed 
  69.  */ 
  70.  public static function zipFile($sourceFile$distFile
  71.  { 
  72.  $zip = new \ZipArchive(); 
  73.  if ($zip->open($distFile, \ZipArchive::CREATE) !== true) { 
  74.   return $sourceFile
  75.  } 
  76.  
  77.  $zip->open($distFile, \ZipArchive::CREATE); 
  78.  foreach ($sourceFile as $file) { 
  79.   $fileContent = file_get_contents($file); 
  80.   $file = iconv('utf-8''GBK'basename($file)); 
  81.   $zip->addFromString($file$fileContent); 
  82.  } 
  83.  $zip->close(); 
  84.  return $distFile
  85.  } 
  86.    
  87.  /** 
  88.  * 下载文件 
  89.  * @param $filePath 
  90.  * @param $fileName 
  91.  */ 
  92.  public static function download($filePath$fileName
  93.  { 
  94.  if (!file_exists($filePath . $fileName)) { 
  95.   header('HTTP/1.1 404 NOT FOUND'); 
  96.  } else { 
  97.   //以只读和二进制模式打开文件 
  98.   $file = fopen($filePath . $fileName"rb"); 
  99.  
  100.   //告诉浏览器这是一个文件流格式的文件 
  101.   Header("Content-type: application/octet-stream"); 
  102.   //请求范围的度量单位 
  103.   Header("Accept-Ranges: bytes"); 
  104.   //Content-Length是指定包含于请求或响应中数据的字节长度 
  105.   Header("Accept-Length: " . filesize($filePath . $fileName)); 
  106.   //用来告诉浏览器,文件是可以当做附件被下载,下载后的文件名称为$file_name该变量的值 
  107.   Header("Content-Disposition: attachment; filename=" . $fileName); 
  108.  
  109.   //读取文件内容并直接输出到浏览器 
  110.   echo fread($filefilesize($filePath . $fileName)); 
  111.   fclose($file); 
  112.   exit(); 
  113.  } 
  114.  } 

调用处代码:

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

Tags: PHP内存溢出

分享到: