PHP实现实时生成并下载超大数据量的EXCEL文件详解


Posted in PHP onOctober 23, 2017

前言

最近在工作中接到一个需求,通过选择的时间段导出对应的用户访问日志到excel中, 由于用户量较大,经常会有导出50万加数据的情况。而常用的PHPexcel包需要把所有数据拿到后才能生成excel, 在面对生成超大数据量的excel文件时这显然是会造成内存溢出的,所以考虑使用让PHP边写入输出流边让浏览器下载的形式来完成需求。

我们通过如下的方式写入PHP输出流

$fp = fopen('php://output', 'a');
fputs($fp, 'strings');
....
....
fclose($fp)

php://output是一个可写的输出流,允许程序像操作文件一样将输出写入到输出流中,PHP会把输出流中的内容发送给web服务器并返回给发起请求的浏览器

另外由于excel数据是从数据库里逐步读出然后写入输出流的所以需要将PHP的执行时间设长一点(默认30秒)set_time_limit(0)不对PHP执行时间做限制。

注:以下代码只是阐明生成大数据量EXCEL的思路和步骤,并且在去掉项目业务代码后程序有语法错误不能拿来直接运行,请根据自己的需求填充对应的业务代码!

/**
  * 文章访问日志
  * 下载的日志文件通常很大, 所以先设置csv相关的Header头, 然后打开
  * PHP output流, 渐进式的往output流中写入数据, 写到一定量后将系统缓冲冲刷到响应中
  * 避免缓冲溢出
  */
 public function articleAccessLog($timeStart, $timeEnd)
 {
  set_time_limit(0);
  $columns = [
   '文章ID', '文章标题', ......
  ];
  $csvFileName = '用户日志' . $timeStart .'_'. $timeEnd . '.xlsx';
  //设置好告诉浏览器要下载excel文件的headers
  header('Content-Description: File Transfer');
  header('Content-Type: application/vnd.ms-excel');
  header('Content-Disposition: attachment; filename="'. $fileName .'"');
  header('Expires: 0');
  header('Cache-Control: must-revalidate');
  header('Pragma: public');
  $fp = fopen('php://output', 'a');//打开output流
  mb_convert_variables('GBK', 'UTF-8', $columns);
  fputcsv($fp, $columns);//将数据格式化为CSV格式并写入到output流中
  $accessNum = '1000000'//从数据库获取总量,假设是一百万
  $perSize = 1000;//每次查询的条数
  $pages = ceil($accessNum / $perSize);
  $lastId = 0;
  for($i = 1; $i <= $pages; $i++) {
   $accessLog = $logService->getArticleAccessLog($timeStart, $timeEnd, $lastId, $perSize);
   foreach($accessLog as $access) {
    $rowData = [
     ......//每一行的数据
    ];
    mb_convert_variables('GBK', 'UTF-8', $rowData);
    fputcsv($fp, $rowData);
    $lastId = $access->id;
   }
   unset($accessLog);//释放变量的内存
   //刷新输出缓冲到浏览器
   ob_flush();
   flush();//必须同时使用 ob_flush() 和flush() 函数来刷新输出缓冲。
  }
  fclose($fp);
  exit();
 }

好了, 其实很简单,就是用逐步写入输出流并发送到浏览器让浏览器去逐步下载整个文件,由于是逐步写入的无法获取文件的总体size所以就没办法通过设置header("Content-Length: $size");在下载前告诉浏览器这个文件有多大了。不过不影响整体的效果这里的核心问题是解决大文件的实时生成和下载。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

PHP 相关文章推荐
开发大型PHP项目的方法
Oct 09 PHP
我常用的几个类
Oct 09 PHP
PHP VS ASP
Oct 09 PHP
IIS环境下快速安装、配置和调试PHP5.2.0
Dec 17 PHP
PHP获取http请求的头信息实现步骤
Dec 16 PHP
解析php curl_setopt 函数的相关应用及介绍
Jun 17 PHP
使用HMAC-SHA1签名方法详解
Jun 26 PHP
php判断两个浮点数是否相等的方法
Mar 14 PHP
php生成高清缩略图实例详解
Dec 07 PHP
在Mac OS上编译安装Nginx+PHP+MariaDB开发环境的教程
Feb 23 PHP
php unicode编码和字符串互转的方法
Aug 12 PHP
php正则表达式基本知识与应用详解【经典教程】
Apr 17 PHP
Laravel学习教程之model validation的使用示例
Oct 23 #PHP
laravel实现批量更新多条记录的方法示例
Oct 22 #PHP
利用PHP获取汉字首字母并且分组排序详解
Oct 22 #PHP
Laravel 5.4因特殊字段太长导致migrations报错的解决
Oct 22 #PHP
PHP高效获取远程图片尺寸和大小的实现方法
Oct 20 #PHP
PHP静态延迟绑定和普通静态效率的对比
Oct 20 #PHP
php+ajax实现仿百度查询下拉内容功能示例
Oct 20 #PHP
You might like
神族 Protoss 剧情介绍
2020/03/14 星际争霸
PHP 如何向 MySQL 发送数据
2006/10/09 PHP
PHP图像处理类库及演示分享
2015/05/17 PHP
JavaScript 继承详解(一)
2009/07/13 Javascript
jquery特效 幻灯片效果示例代码
2013/07/16 Javascript
基于jquery的文章中所有图片width大小批量设置方法
2013/08/01 Javascript
JS小功能(checkbox实现全选和全取消)实例代码
2013/11/28 Javascript
javascript浏览器窗口之间传递数据的方法
2015/01/20 Javascript
JS实现横向拉伸动感伸缩菜单效果代码
2015/09/04 Javascript
JS实现3D图片旋转展示效果代码
2015/09/22 Javascript
javascript省市级联功能实现方法实例详解
2015/10/20 Javascript
开启Javascript中apply、call、bind的用法之旅模式
2015/10/28 Javascript
JS遍历数组和对象的区别及递归遍历对象、数组、属性的方法详解
2016/06/14 Javascript
Angular2中Bootstrap界面库ng-bootstrap详解
2016/10/18 Javascript
jQuery实现简易的输入框字数计数功能示例
2017/01/16 Javascript
javascript实现右下角广告框效果
2017/02/01 Javascript
angularjs实现分页和搜索功能
2018/01/03 Javascript
echarts同一页面中四个图表切换的js数据交互方法示例
2018/07/03 Javascript
详解如何在webpack中做预渲染降低首屏空白时间
2018/08/22 Javascript
js代码规范之Eslint安装与配置详解
2018/09/08 Javascript
微信小程序实现单个或多个倒计时功能
2020/11/01 Javascript
Python编程之序列操作实例详解
2017/07/22 Python
selenium+headless chrome爬虫的实现示例
2021/01/08 Python
python-jwt用户认证食用教学的实现方法
2021/01/19 Python
在SQL Server中创建数据库主要有那种方式
2013/09/10 面试题
.net笔试题
2014/03/03 面试题
应届生高等护理求职信
2013/10/12 职场文书
给老师的道歉信
2014/01/11 职场文书
人力资源本科毕业生求职信
2014/06/04 职场文书
村主任“四风”问题个人对照检查材料思想汇报
2014/10/02 职场文书
原料仓管员岗位职责
2015/04/01 职场文书
房地产销售助理岗位职责
2015/04/14 职场文书
幼儿园六一儿童节主持词
2015/06/30 职场文书
2016党风廉政建设心得体会范文
2016/01/25 职场文书
python批量创建变量并赋值操作
2021/06/03 Python
用JS实现飞机大战小游戏
2021/06/09 Javascript