fastadmin-PHP-导出少量数据PhpOffice以及百万级别数据csv压缩

在进行数据导出的时候,少量的数据可以使用phpexcel,但大量的数据用phpexcel就很消耗资源了。

在使用fastadmin做数据导出的时候,相关的代码请参考:https://blog.csdn.net/bingyu709/article/details/141949034

我自己这边因为数据量会很大,所以代码层做了一个数量的划分,少于50000走phpexcel导出,大于50000走csv,并且打包zip。

并且我的逻辑是导出成功后把文件防止项目指定目录下,并没有做一个直接下来,如果想直接从浏览器下载请自己网上找其他代码参考,我所参考的帖子是直接下载的,大家可以试试看。

public function export() {
        if ($this->request->isPost()) {



            $filter             = $this->request->request("filter", '');
            $filter_arr         = (array)json_decode($filter, true);

            $ids                = $this->request->request('ids','');

            $scene_id                   = CService::MANAGE_ALL_CUSTOMER;
            if (!empty($filter_arr) && isset($filter_arr['scene_id']))  {
                $scene_id       = intval($filter_arr['scene_id']);
            }

            $scene_where        = $this->getFilterAllSceneWhere($scene_id);

            list($where, $sort, $order, $offset, $limit) = $this->buildAllParams();

            if (!empty($ids) && $ids != '' && $ids != 'all') {
                $ids_arr        = explode(',',$ids);
                $where['id']    = ['in', $ids_arr];
            }

            $where              = array_merge($scene_where,$where);


            $count              = CModel::model()->where($where)->count();


            if ($count > 0) {

                if ($count > 50000) {

                    set_time_limit(0);
                    ini_set('memory_limit', '128M');

                    $xlsTitle = [   '字段1', '字段2', '字段3'
                                ];

                    // buffer计数器
                    $cnt = 0;



                    $fileName           = '客户导出'.date('YmdHis').rand(10000000,88888888);//文件名称
                    $pro_path           = ExportService::EXPORT_CUSTOMER_FILE_PATH; // 项目目录
                    $fileName           = $pro_path . $fileName;

                    $zip_file_name      = $fileName.'.zip';

                    // 输出Excel文件头,可把user.csv换成你要的文件名
                    header('Content-Type: application/vnd.ms-excel;charset=utf-8');
                    header('Content-Disposition: attachment;filename="' . $zip_file_name . '"');
                    header('Cache-Control: max-age=0');

                    $fileNameArr = [];

                    $every_step_limit       = 10000;
                    $loop_number_count      = intval(ceil($count / $every_step_limit));


                    for ($i = 0; $i < $loop_number_count; $i ++) {


                        $fp = fopen($fileName . '_' . ($i+1) . '.csv', 'w'); //生成临时文件
                        fwrite($fp, chr(0xEF).chr(0xBB).chr(0xBF));/*防止乱码  */

                        // chmod('attack_ip_info_' . $i . '.csv',777);//修改可执行权限
                        $fileNameArr[] = $fileName . '_' . ($i+1) . '.csv'; // 将数据通过fputcsv写到文件句柄
                        fputcsv($fp, $xlsTitle);
                        /******************** 调整位置开始 ***************************/

                        $limit          = $i * $every_step_limit;
                        $limit_list     = ExportService::service()->getExportList($where, $limit, $every_step_limit);
                        $list           = ExportService::service()->formatExportData($limit_list, $this->auth_global_search_organise_id);
                        if (!empty($list)) {
                            foreach ($list as $item) {
                                $cnt++;
                                if ($limit == $cnt) {
                                    // 刷新一下输出buffer,防止由于数据过多造成问题
                                    ob_flush();
                                    flush();
                                    $cnt = 0;
                                }
                                fputcsv($fp, $item);
                            }
                            unset($list);
                        }
                        fclose($fp); // 每生成一个文件关闭
                    }



                    try {
                        // 进行多个文件压缩
                        $zip = new \ZipArchive();

                        $zip->open($zip_file_name, $zip::CREATE); // 打开压缩包
                        foreach ($fileNameArr as $file) {
                            $zip->addFile($file, basename($file)); // 向压缩包中添加文件
                        }
                        $zip->close();  // 关闭压缩包
                        foreach ($fileNameArr as $file) {
                            unlink($file); // 删除csv临时文件
                        }

                        if (file_exists($zip_file_name)) {
                           
                            $this->success('操作成功');
                        } else {
                            $this->error('操作失败');
                        }
                    } catch (ValidateException|PDOException|Exception $e) {
                        $this->error('操作失败');
                    }
                } else {
                    set_time_limit(0);

                    $spreadsheet    = new Spreadsheet();
                    $sheet          = $spreadsheet->getActiveSheet();

                    $sheet->getStyle('A:S')->getAlignment()
                        ->setVertical(\PhpOffice\PhpSpreadsheet\Style\Alignment::VERTICAL_CENTER) //设置垂直居中
                        ->setHorizontal(\PhpOffice\PhpSpreadsheet\Style\Alignment::HORIZONTAL_CENTER) //设置水平居中
                        ->setWrapText(true); //设置自动换行

                    $titleStyleArray     = [
                        'font'      => [
                            'name'  => 'SimHei', // 中文字体,"黑体"的类似字体
                            'bold'  => true,
                            'size'  => 12,
                        ]
                    ];
                    $sheet->getStyle('A1:Z1')->applyFromArray($titleStyleArray);

                    $lineStyleArray     = [
                        'font'      => [
                            'name'  => 'SimHei', // 中文字体,"黑体"的类似字体
                        ]
                    ];

                    //设置表头
                    $sheet->setCellValue('A1', '字段1');
                    $sheet->setCellValue('B1', '字段2');
                    $sheet->setCellValue('C1', '字段3');
            .......
$sheet->getColumnDimension('A')->setWidth(15); $sheet->getColumnDimension('B')->setWidth(15); $sheet->getColumnDimension('C')->setWidth(15);
            .......
//$sheet->getDefaultRowDimension()->setRowHeight(60); //设置行高 $res_list = ExportService::service()->getExportCustomerList($where); $list = ExportService::service()->formatExportCustomerData($res_list, $this->auth_global_search_organise_id); if (!empty($list)) { //输出表格 foreach($list as $keys => $value) { $keys = $keys + 2; $sheet->setCellValue('A'.$keys,$value['field1'] ?? '')->getStyle('A')->applyFromArray($lineStyleArray); $sheet->setCellValue('B'.$keys,$value['field2'] ?? '')->getStyle('B')->applyFromArray($lineStyleArray); $sheet->setCellValue('C'.$keys,$value['field3'] ?? '')->getStyle('C')->applyFromArray($lineStyleArray);
                 .........
} } // 保存Excel文件到服务器的指定路径 $pro_path = ExportService::EXPORT_CUSTOMER_FILE_PATH; // 项目目录 $file_name = '导出'.date('YmdHis').rand(10000000,88888888).'.xlsx'; $file_path = $pro_path . $file_name; // if (!file_exists($file_path)) { // fopen($file_path, 'w'); // } try { $writer = \PhpOffice\PhpSpreadsheet\IOFactory::createWriter($spreadsheet, 'Xlsx'); $writer->save($file_path); if (file_exists($file_path)) { $this->success('操作成功'); } else { $this->error('操作失败'); } } catch (ValidateException|PDOException|Exception $e) { $this->error('操作失败'); } } } } }

 

posted @ 2024-11-19 12:10  温柔的风  阅读(52)  评论(0编辑  收藏  举报