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 相关文章推荐
收集的DedeCMS一些使用经验
Mar 17 PHP
ThinkPHP php 框架学习笔记
Oct 30 PHP
使用NetBeans + Xdebug调试PHP程序的方法
Apr 12 PHP
基于php缓存的详解
May 15 PHP
PHPer 需要了解的 5 个 Composer 小技巧
Aug 18 PHP
php类的扩展和继承用法实例
Jun 20 PHP
php实现可逆加密的方法
Aug 11 PHP
round robin权重轮循算法php实现代码
May 28 PHP
php获取linux命令结果的实例
Mar 13 PHP
PHP preg_match实现正则表达式匹配功能【输出是否匹配及匹配值】
Jul 19 PHP
laravel5.2表单验证,并显示错误信息的实例
Sep 29 PHP
如何在Mac上通过docker配置PHP开发环境
May 29 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
DC的38部超级英雄动画电影
2020/03/03 欧美动漫
初步介绍PHP扩展开发经验分享
2012/09/06 PHP
phpstudy2018升级MySQL5.5为5.7教程(图文)
2018/10/24 PHP
IE浏览器兼容Firefox的JS脚本的代码
2008/10/23 Javascript
google jQuery 引用文件,jQuery 引用地址集合(jquery 1.2.6至jquery1.5.2)
2011/04/24 Javascript
JS父页面与子页面相互传值方法
2014/03/05 Javascript
JavaScript中的console.log()函数详细介绍
2014/12/29 Javascript
老司机带你解读jQuery插件开发流程
2016/05/16 Javascript
jQuery获取当前点击的对象元素(实现代码)
2016/05/19 Javascript
详解weex默认webpack.config.js改造
2018/01/08 Javascript
浅谈Vue网络请求之interceptors实际应用
2018/02/28 Javascript
Vue写一个简单的倒计时按钮功能
2018/04/20 Javascript
JS代码屏蔽F12,右键,粘贴,复制,剪切,选中,操作实例
2019/09/17 Javascript
JavaScript this关键字的深入详解
2021/01/14 Javascript
[00:32]DOTA2上海特级锦标赛 COL战队宣传片
2016/03/04 DOTA
[02:26]2016国际邀请赛8月3日开战 中国军团出征西雅图
2016/08/02 DOTA
使用Python的web.py框架实现类似Django的ORM查询的教程
2015/05/02 Python
python虚拟环境virtualenv的使用教程
2017/10/20 Python
pandas 选择某几列的方法
2018/07/03 Python
关于PyTorch源码解读之torchvision.models
2019/08/17 Python
Python安装与卸载流程详细步骤(图解)
2020/02/20 Python
用CSS3实现Win8风格的方格导航菜单效果
2013/04/10 HTML / CSS
HTML5离线缓存Manifest是什么
2016/03/09 HTML / CSS
Html5实现如何在两个div元素之间拖放图像
2013/03/29 HTML / CSS
巴西在线鞋店:Shoestock
2017/10/28 全球购物
英国最大的香水商店:The Fragrance Shop
2018/07/06 全球购物
美国儿童玩具、装扮和玩偶商店:Magic Cabin
2018/09/02 全球购物
住房公积金接收函
2014/01/09 职场文书
公司周年庆活动方案
2014/08/25 职场文书
2015年青年志愿者协会工作总结
2015/04/27 职场文书
信用卡工资证明范本
2015/06/19 职场文书
货款欠条范本
2015/07/03 职场文书
2019暑假学生安全口号
2019/06/27 职场文书
职工的安全责任书范文!
2019/07/02 职场文书
Mysql实现主从配置和多主多从配置
2021/06/02 MySQL
Mysql数据库按时间点恢复实战记录
2021/06/30 MySQL