详解PHP用xlswriter优化Excel导出性能

 3434

本篇文章给大家带来了关于php的相关知识,其中主要跟大家聊一聊xlswriter扩展是什么?怎么使用xlswriter扩展优化Excel导出性能,感兴趣的朋友下面一起来看一下吧,希望对大家有帮助。

关于xlswriter

xlswriter 是一个 PHP C 扩展,旨在提升php在导出大数据量时的性能问题,支持 windows / Linux 。可用于在 Excel 2007+ XLSX 文件中读取数据,插入多个工作表,写入文本、数字、公式、日期、图表、图片和超链接。

它具备以下特性:

一、写入

100%兼容的 Excel XLSX 文件

完整的 Excel 格式

合并单元格

定义工作表名称

过滤器

图表

数据验证和下拉列表

工作表 PNG/JPEG 图像

用于写入大文件的内存优化模式

适用于 Linux,FreeBSD,OpenBSD,OS X,Windows

编译为 32 位和 64 位

FreeBSD 许可证

唯一的依赖是 zlib

二、读取

完整读取数据

光标读取数据

按数据类型读取

xlsx 转 CSV

性能对比

先感谢网友提供数据

下载安装

github源码

https://github.com/viest/php-ext-xlswriter

xlswriter 文档

https://xlswriter-docs.viest.me/zh-cn/an-zhuang/huan-jing-yao-qiu

下载 ide helper

  1. composer require viest/php-ext-xlswriter-ide-helper:dev-master

但是我一直下载失败,于是去github仓库直接下载 https://github.com/viest/php-ext-xlswriter-ide-helper
然后将里面的几个类复制到一个 xlswriter_ide_helper.php 文件里面,将这个文件放到你的项目中就有代码提示了。

安装 xlswriter 扩展

此处在docker中安装

  1. docker exec -it php72-fpm bashcd /usr/local/bin
  2. pecl install xlswriter
  3. docker-php-ext-enable xlswriter
  4. php -m
  5.  
  6. php --ri xlswriter
  7. Version => 1.3.6
  8.  
  9. docker restart php72-fpm

性能测试:

测试数据:20 列,每列长度为 19 英文字母

Xlswriter


详解PHP用xlswriter优化Excel导出性能


PHPSpreadSheet


详解PHP用xlswriter优化Excel导出性能


PHP_XLSXWriter


详解PHP用xlswriter优化Excel导出性能


使用示例:

  1. private function rankPersonExport($activityInfo, $list){
  2.     $date = date('Y-m-d');
  3.     $filename = "{$activityInfo['orgname']}-{$activityInfo['name']}-个人排行榜-{$date}";
  4.     $header = ['名次', '用户ID', '对接账号', '姓名', '电话', '部门ID', '一级部门', '二级部门', '三级部门', '总积分', '最后积分时间', "毫秒"];
  5.     if (!empty($activityInfo['ext'])) {
  6.         $extArr = json_decode($activityInfo['ext'], true);
  7.         foreach ($extArr as $errItem) {
  8.             array_push($header, $errItem['name']);
  9.         }
  10.     }
  11.     // list
  12.     $listVal = [];
  13.     foreach($list as $v){
  14.         $temp = [
  15.             $v['rank'],
  16.             $v['userid'],
  17.             $v['userName'],
  18.             $v['nickName'],
  19.             $v['phone'],
  20.             $v['departid'],
  21.             $v['topDepartName'],
  22.             $v['secDepartName'],
  23.             $v['thirdDepartName'],
  24.             $v['score'],
  25.             $v['updatetime'],
  26.             $v['micro'],
  27.         ];
  28.  
  29.         if (!empty($v['ext'])) {
  30.             $extArr = explode('|', $v['ext']);
  31.             foreach ($extArr as $k2 => $v2) {
  32.                 $errItemArr = explode('^', $v2);
  33.                 array_push($temp, $errItemArr[1]);
  34.             }
  35.         }
  36.         array_push($listVal, $temp);
  37.     }
  38.  
  39.     $re = downloadXLSX($filename, $header, $listVal);
  40.     if($re){
  41.         return $this->output(0, $re);
  42.     }else{
  43.         return $this->output(1, 'success');
  44.     }
  45. }
  1. function getTmpDir(): string{
  2.     $tmp = ini_get('upload_tmp_dir');
  3.  
  4.     if ($tmp !== False && file_exists($tmp)) {
  5.         return realpath($tmp);
  6.     }
  7.  
  8.     return realpath(sys_get_temp_dir());}/**
  9.  * download xlsx file
  10.  *
  11.  * @param string $filename
  12.  * @param array $header
  13.  * @param array $list
  14.  * @return string errmsg
  15.  */function downloadXLSX(string $filename, array $header, array $list): string{
  16.     try {
  17.         $config = ['path' => getTmpDir() . '/'];
  18.         $excel  = (new \Vtiful\Kernel\Excel($config))->fileName($filename.'.xlsx', 'Sheet1');
  19.         $fileHandle = $excel->getHandle();
  20.         $format1    = new \Vtiful\Kernel\Format($fileHandle);
  21.         $format2    = new \Vtiful\Kernel\Format($fileHandle);
  22.  
  23.         // title style
  24.         $titleStyle = $format1->fontSize(16)
  25.             ->bold()
  26.             ->font("Calibri")
  27.             ->align(\Vtiful\Kernel\Format::FORMAT_ALIGN_CENTER, \Vtiful\Kernel\Format::FORMAT_ALIGN_VERTICAL_CENTER)
  28.             ->toResource();
  29.  
  30.         // global style
  31.         $globalStyle = $format2->fontSize(10)
  32.             ->font("Calibri")
  33.             ->align(\Vtiful\Kernel\Format::FORMAT_ALIGN_CENTER, \Vtiful\Kernel\Format::FORMAT_ALIGN_VERTICAL_CENTER)
  34.             ->border(\Vtiful\Kernel\Format::BORDER_THIN)
  35.             ->toResource();
  36.  
  37.         $headerLen = count($header);
  38.  
  39.         // header
  40.         array_unshift($list, $header);
  41.  
  42.         // title
  43.         $title = array_fill(1, $headerLen - 1, '');
  44.         $title[0] = $filename;
  45.         array_unshift($list, $title);
  46.  
  47.         $end = strtoupper(chr(65 + $headerLen - 1));
  48.         // column style
  49.         $excel->setColumn("A:{$end}", 15, $globalStyle);
  50.         // title
  51.         $excel->MergeCells("A1:{$end}1", $filename)->setRow("A1", 25, $titleStyle);
  52.         // 冻结前两行,列不冻结
  53.         $excel->freezePanes(2, 0);
  54.         // 数据
  55.         $filePath = $excel->data($list)->output();
  56.  
  57.         header("Content-Disposition:attachment;filename={$filename}.xlsx");
  58.  
  59.         $re = copy($filePath, 'php://output');
  60.         if ($re === false) {
  61.             $err = 'failed to write output';
  62.         } else {
  63.             $err = '';
  64.         }
  65.         @unlink($filePath);
  66.  
  67.         return $err;
  68.     } catch (\Vtiful\Kernel\Exception $e) {
  69.         return $e->getMessage();
  70.     }
  71. }

如果发现下载的文件有时候打不开,那应该是你使用了官方的DEMO,问题出在 filesize(),这个函数是有缓存的,所以你会发现下载下来的文件和原始的文件大小不一样。要么像我一样不去设置 Content-Length,要么使用 clearstatcache()手动清除缓存。

实测5w条记录导出耗时1.5s,效果还是很强劲的。


导出效果


详解PHP用xlswriter优化Excel导出性能


本文网址:https://www.zztuku.com/index.php/detail-13891.html
站长图库 - 详解PHP用xlswriter优化Excel导出性能
申明:本文转载于《CSDN》,如有侵犯,请 联系我们 删除。

评论(0)条

您还没有登录,请 登录 后发表评论!

提示:请勿发布广告垃圾评论,否则封号处理!!

    编辑推荐