本篇文章給大家?guī)砹岁P(guān)于php的相關(guān)知識,其中主要跟大家聊一聊xlswriter擴展是什么?怎么使用xlswriter擴展優(yōu)化Excel導(dǎo)出性能,感興趣的朋友下面一起來看一下吧,希望對大家有幫助。
關(guān)于xlswriter
xlswriter 是一個 PHP C 擴展,旨在提升php在導(dǎo)出大數(shù)據(jù)量時的性能問題,支持 windows / Linux 。可用于在 Excel 2007+ XLSX 文件中讀取數(shù)據(jù),插入多個工作表,寫入文本、數(shù)字、公式、日期、圖表、圖片和超鏈接。
它具備以下特性:
一、寫入
100%兼容的 Excel XLSX 文件
完整的 Excel 格式
合并單元格
定義工作表名稱
過濾器
圖表
數(shù)據(jù)驗證和下拉列表
工作表 PNG/JPEG 圖像
用于寫入大文件的內(nèi)存優(yōu)化模式
適用于 Linux,F(xiàn)reeBSD,OpenBSD,OS X,Windows
編譯為 32 位和 64 位
FreeBSD 許可證
唯一的依賴是 zlib
二、讀取
完整讀取數(shù)據(jù)
光標(biāo)讀取數(shù)據(jù)
按數(shù)據(jù)類型讀取
xlsx 轉(zhuǎn) CSV
性能對比
先感謝網(wǎng)友提供數(shù)據(jù)
下載安裝
github源碼
https://github.com/viest/php-ext-xlswriter
xlswriter 文檔
https://xlswriter-docs.viest.me/zh-cn/an-zhuang/huan-jing-yao-qiu
下載 ide helper
composer require viest/php-ext-xlswriter-ide-helper:dev-master
但是我一直下載失敗,于是去github倉庫直接下載 https://github.com/viest/php-ext-xlswriter-ide-helper
然后將里面的幾個類復(fù)制到一個 xlswriter_ide_helper.php 文件里面,將這個文件放到你的項目中就有代碼提示了。
安裝 xlswriter 擴展
此處在docker中安裝
docker exec -it php72-fpm bashcd /usr/local/bin pecl install xlswriter docker-php-ext-enable xlswriter php -m php --ri xlswriter Version => 1.3.6 docker restart php72-fpm
性能測試:
測試數(shù)據(jù):20 列,每列長度為 19 英文字母
Xlswriter
PHPSpreadSheet
PHP_XLSXWriter
使用示例:
private function rankPersonExport($activityInfo, $list){ $date = date('Y-m-d'); $filename = "{$activityInfo['orgname']}-{$activityInfo['name']}-個人排行榜-{$date}"; $header = ['名次', '用戶ID', '對接賬號', '姓名', '電話', '部門ID', '一級部門', '二級部門', '三級部門', '總積分', '最后積分時間', "毫秒"]; if (!empty($activityInfo['ext'])) { $extArr = json_decode($activityInfo['ext'], true); foreach ($extArr as $errItem) { array_push($header, $errItem['name']); } } // list $listVal = []; foreach($list as $v){ $temp = [ $v['rank'], $v['userid'], $v['userName'], $v['nickName'], $v['phone'], $v['departid'], $v['topDepartName'], $v['secDepartName'], $v['thirdDepartName'], $v['score'], $v['updatetime'], $v['micro'], ]; if (!empty($v['ext'])) { $extArr = explode('|', $v['ext']); foreach ($extArr as $k2 => $v2) { $errItemArr = explode('^', $v2); array_push($temp, $errItemArr[1]); } } array_push($listVal, $temp); } $re = downloadXLSX($filename, $header, $listVal); if($re){ return $this->output(0, $re); }else{ return $this->output(1, 'success'); } }
function getTmpDir(): string{ $tmp = ini_get('upload_tmp_dir'); if ($tmp !== False && file_exists($tmp)) { return realpath($tmp); } return realpath(sys_get_temp_dir());}/** * download xlsx file * * @param string $filename * @param array $header * @param array $list * @return string errmsg */function downloadXLSX(string $filename, array $header, array $list): string{ try { $config = ['path' => getTmpDir() . '/']; $excel = (new \Vtiful\Kernel\Excel($config))->fileName($filename.'.xlsx', 'Sheet1'); $fileHandle = $excel->getHandle(); $format1 = new \Vtiful\Kernel\Format($fileHandle); $format2 = new \Vtiful\Kernel\Format($fileHandle); // title style $titleStyle = $format1->fontSize(16) ->bold() ->font("Calibri") ->align(\Vtiful\Kernel\Format::FORMAT_ALIGN_CENTER, \Vtiful\Kernel\Format::FORMAT_ALIGN_VERTICAL_CENTER) ->toResource(); // global style $globalStyle = $format2->fontSize(10) ->font("Calibri") ->align(\Vtiful\Kernel\Format::FORMAT_ALIGN_CENTER, \Vtiful\Kernel\Format::FORMAT_ALIGN_VERTICAL_CENTER) ->border(\Vtiful\Kernel\Format::BORDER_THIN) ->toResource(); $headerLen = count($header); // header array_unshift($list, $header); // title $title = array_fill(1, $headerLen - 1, ''); $title[0] = $filename; array_unshift($list, $title); $end = strtoupper(chr(65 + $headerLen - 1)); // column style $excel->setColumn("A:{$end}", 15, $globalStyle); // title $excel->MergeCells("A1:{$end}1", $filename)->setRow("A1", 25, $titleStyle); // 凍結(jié)前兩行,列不凍結(jié) $excel->freezePanes(2, 0); // 數(shù)據(jù) $filePath = $excel->data($list)->output(); header("Content-Disposition:attachment;filename={$filename}.xlsx"); $re = copy($filePath, 'php://output'); if ($re === false) { $err = 'failed to write output'; } else { $err = ''; } @unlink($filePath); return $err; } catch (\Vtiful\Kernel\Exception $e) { return $e->getMessage(); } }
如果發(fā)現(xiàn)下載的文件有時候打不開,那應(yīng)該是你使用了官方的DEMO,問題出在 filesize(),這個函數(shù)是有緩存的,所以你會發(fā)現(xiàn)下載下來的文件和原始的文件大小不一樣。要么像我一樣不去設(shè)置 Content-Length,要么使用 clearstatcache()手動清除緩存。
實測5w條記錄導(dǎo)出耗時1.5s,效果還是很強勁的。
導(dǎo)出效果