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获取远程图片并把它保存到本地的代码
Apr 07 PHP
php 移除数组重复元素的一点说明
Nov 27 PHP
PHP 数组入门教程小结
May 20 PHP
php数组函数序列之in_array() 查找数组值是否存在
Oct 29 PHP
PHP中文处理 中文字符串截取(mb_substr)和获取中文字符串字数
Nov 10 PHP
PHP写的求多项式导数的函数代码
Jul 04 PHP
使用Discuz关键词服务器实现PHP中文分词
Mar 11 PHP
PHP错误Warning: Cannot modify header information - headers already sent by解决方法
Sep 27 PHP
PHP Socket网络操作类定义与用法示例
Aug 30 PHP
PHP笛卡尔积实现算法示例
Jul 30 PHP
php生成word并下载代码实例
Mar 15 PHP
YII2框架中使用RBAC对模块,控制器,方法的权限控制及规则的使用示例
Mar 18 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
PHP程序员面试 切忌急功近利(更需要注重以后的发展)
2010/09/01 PHP
PHP中call_user_func_array()函数的用法演示
2012/02/05 PHP
php实现URL加密解密的方法
2016/11/17 PHP
PHP实现正则表达式分组捕获操作示例
2018/02/03 PHP
Javascript模板技术
2007/04/27 Javascript
javascript数组的扩展实现代码集合
2008/06/01 Javascript
Javascript结合css实现网页换肤功能
2009/11/02 Javascript
两个listbox实现选项的添加删除和搜索
2013/03/01 Javascript
Vue-router路由判断页面未登录跳转到登录页面的实例
2017/10/26 Javascript
js 取消页面可以选中文字的功能方法
2018/01/02 Javascript
浅析node应用的timing-attack安全漏洞
2018/02/28 Javascript
一种angular的方法级的缓存注解(装饰器)
2018/03/13 Javascript
小程序click-scroll组件设计
2019/06/18 Javascript
react项目从新建到部署的实现示例
2021/02/19 Javascript
[37:02]OG vs INfamous 2019国际邀请赛小组赛 BO2 第二场 8.15
2019/08/17 DOTA
python实现调用其他python脚本的方法
2014/10/05 Python
win系统下为Python3.5安装flask-mongoengine 库
2016/12/20 Python
Django如何实现内容缓存示例详解
2017/09/24 Python
flask中过滤器的使用详解
2018/08/01 Python
python实现大文件分割与合并
2019/07/22 Python
html5 touch事件实现触屏页面上下滑动(一)
2016/03/10 HTML / CSS
基于HTML5超酷摄像头(HTML5 webcam)拍照功能实现代码
2012/12/13 HTML / CSS
美国在线鲜花速递:ProFlowers
2017/01/05 全球购物
Boden澳大利亚官网:英国在线服装公司
2018/08/05 全球购物
戴森西班牙官网:Dyson西班牙
2020/02/04 全球购物
澳大利亚香水在线商店:City Perfume
2020/09/02 全球购物
Diesel美国网上商店:意大利牛仔时装品牌
2020/12/10 全球购物
高中生学习总结的自我评价范文
2013/10/13 职场文书
管道维修工岗位职责
2013/12/27 职场文书
父亲生日宴会答谢词
2014/01/10 职场文书
英文商务邀请信
2014/01/22 职场文书
求职简历中自我评价
2014/01/28 职场文书
财务负责人任命书
2014/06/06 职场文书
以幸福为主题的活动方案
2014/08/22 职场文书
本科毕业论文导师评语
2014/12/31 职场文书
护士辞职信怎么写
2015/02/27 职场文书