php - 将大量数据导出到 xlsx

标签 php laravel laravel-5 phpexcel laravel-excel

我需要使用 MYISAM 引擎将一个巨大的数据集从 MySQL 数据库表导出到 Laravel 中的 .xlsx 文件中。

我正在使用 maatwebsite/laravel-excel包,基于 PHPExcel .

数据集包含大约 500,000 行和 93 列(大约 46,500,000 个单元格),以及大量关于标题结构的计算。

这是我目前使用的代码:

// $excel_data contains some data regarding the project, nothing relevant here
$output = Excel::create('myproject-' . $excel_data->project->name . '-'.date('Y-m-d H:i:s') . '-export', function($excel) use($excel_data) {

        // Set the title
        $excel->setTitle($excel_data->project->name . ' Export');

        $excel->sheet('Data', function($sheet) use($excel_data) {

            $rowPointer = 1;

            $query = DB::table('task_metas')
                ->where([
                    ['project_id', '=', $excel_data->project->id],
                    ['deleted_at', '=', null]
                ])
                ->orderBy('id');

            $totalRecords = $query->count();
            // my server can't handle a request that returns more than 20k rows so I am chunking the results in batches of 15000 to be on the safe side
            $query->chunk(15000, function($taskmetas) use($sheet, &$rowPointer, $totalRecords) {
                // Iterate over taskmetas
                foreach ($taskmetas as $taskmeta) {
                    // other columns and header structure omitted for clarity
                    $sheet->setCellValue('A' . $rowPointer, $rowPointer);
                    $sheet->setCellValue('B' . $rowPointer, $taskmeta->id);
                    $sheet->setCellValue('C' . $rowPointer, $taskmeta->url);

                    // Move on to the next row
                    $rowPointer++;
                }
                // logging the progress of the export
                activity()
                    ->log("wrote taskmeta to row " . $rowPointer . "/" . $totalRecords);

                unset($taskmetas);
            });
        });

    });

    $output->download('xlsx');

根据日志,行已成功写入文件,但文件创建本身需要很长时间。事实上,它在 1 小时内没有完成(这是该函数的最长执行时间)。

将它导出到 csv 效果很好,在大约 10 分钟内它会编译文件并下载它,但是我无法使用它 - 输出文件需要xlsx.

如何加快文件创建过程?只要我能达到相同的结果,我也愿意接受其他替代方案。

最佳答案

我有3个建议:

  1. 使用 cursor (虽然直到今天,我还没有发现它是否比 block 更好 - 也许你的案例可以验证这一点) - 真诚地我只使用了这个并且它与 Eloquent 一起使用。

  2. 减小块的大小。我认为在内存中加载 15000 条记录已经是一个问题。

  3. 首先创建excel 文件,然后在工作表上使用rows() 方法追加多行。 (这可能效果不佳,因为它需要一个数组)

关于php - 将大量数据导出到 xlsx,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45493325/

相关文章:

php - 不直观的递增表达式求值

php - 连接到 MySQL 时出现 PDO 错误

PHP开始时间和结束时间问题

php - 在服务器上启用 PDO,还是不使用 Laravel?

laravel - spatie/laravel-medialibrary 如何获取上传图像的尺寸?

php - Laravel 命令-> 选择循环

php - 增加 phpmyadmin 中的文件大小限制

javascript - 在 laravel Blade 中使用 refs 来预先检查复选框

laravel - Laravel 5 中布局内的嵌套布局

forms - Laravel 5 动态表单验证