laravel 大数据分块导出,避免内存溢出

2024-08-20 16:52

本文主要是介绍laravel 大数据分块导出,避免内存溢出,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

laravel 大数据分块导出,避免内存溢出

数据来源:服务层(userService.php)

控制器:UserController

    public function lists(Request $request){$this->service = new UserService();$params = $request->all();$result = $this->service->lists($params);//功能:分页查询数据if(!empty($params['is_down'])) {return Excel::download(new SignContractExport($this->service, $params, 100), '合约.xlsx');}return  $this->successJson($result, "操作成功");}

导出类:SignContractExport

<?phpnamespace App\Exports;use Maatwebsite\Excel\Concerns\FromCollection;
use Maatwebsite\Excel\Concerns\WithHeadings;
use Maatwebsite\Excel\Concerns\ShouldAutoSize;
use Maatwebsite\Excel\Concerns\WithColumnWidths;
use Maatwebsite\Excel\Concerns\WithStyles;
use Modules\Admin\Services\UserService;
use Maatwebsite\Excel\Concerns\WithTitle;class SignContractExport implements FromCollection,WithHeadings,ShouldAutoSize,WithStyles,WithTitle,WithColumnWidths
{protected $batchSize = 100;public $params = [];private array $header = ['ID','姓名'];public $status;protected $service;protected $chunkSize;/*** 导出*/public function __construct(UserService $service, $params,$chunkSize = 1000){$this->status = $params['status'];$this->myHeader($params['status']);$this->service = $service;//业务对象,里面 提供分页查询的方法$this->params = $params;$this->chunkSize = $chunkSize;}public function map($row): array{$itemMap['id'] = $row->id;$itemMap['mobile'] = $row->user_mobile;}public function title():string{return '合约导出';}public function headings($add = []): array{return $this->header;}public function collection(){$data = collect();// 使用 getDataChunked 方法$this->service->getDataChunked($this->params, $this->chunkSize, function ($chunk) use ($data) {foreach ($chunk as $row) {$data->push($this->map($row));}});return $data;}
}

业务层:UserService

    /*** 大数据导出防止内存溢出做分块儿查询*/public function getDataChunked($params, $chunkSize, callable $callback){$totalFetched = 0;// 通过 chunk 的方式来处理数据,每次处理 $chunkSize 条数据do {$params['page'] = $totalFetched / $chunkSize + 1;$params['page_size'] = $chunkSize;// 获取数据(这里重用 lists 方法)$result = $this->lists($params);if (!empty($result['list'])) {//list:列表数据// 回调函数处理每块数据$callback($result['list']);$totalFetched += count($result['list']);} else {break; // 如果没有数据,跳出循环}} while ($totalFetched < $result['total']);//total:总条数}public function lists($params) {$list = User::query()->->paginate($params['page_size']??10,$columns = ['*'], $pageName = 'page', $params['page']??null);return ['page' => $list->currentPage(),'page_size' => $list->perPage(),'total' => $list->total(),'list' => $data,];}

分析与实现
在 Laravel 中处理大数据导出时,直接加载所有数据到内存中可能导致内存溢出问题。为了避免这种情况,你的实现使用了“分块查询”(chunking)的方法,通过分块处理数据,逐块加载和导出,从而有效地管理内存使用。以下是对你的实现如何避免内存溢出的详细分析:

  1. 数据分块的基本思路
    分块查询:通过 getDataChunked 方法,将大数据集分成若干小块(每块 chunkSize 条数据)。每次查询只处理一块数据,处理完后立即释放内存,再进行下一块的处理。这种方式确保了内存中只保留当前处理的那一小部分数据,避免一次性加载所有数据而导致内存溢出。

回调函数的使用:通过传递一个回调函数 callback 来处理每一块查询到的数据块。这样,你可以在导出 Excel 的过程中,逐块地将数据添加到 Excel 文件中,而不是一次性加载所有数据。

  1. 具体的实现步骤
    Step 1: 获取数据总量并初始化分页
    首先,在 getDataChunked 方法中通过 lists 方法获取数据的总数 total。
    通过初始化的 params[‘page’] 和 params[‘page_size’] 来定义分页的大小和当前页数。
    Step 2: 分块查询和处理
    使用 do…while 循环进行数据的分块查询。每次循环中,根据当前已获取的数据量计算出当前页数,然后查询这一页的数据。
    每次查询到的数据块(如 1000 条)通过 callback 回调函数立即处理(在本例中,导出到 Excel 文件中)。
    处理完一块数据后,循环检查是否还需要继续查询更多数据,直到所有数据都被处理完。
    Step 3: 内存释放
    每次处理完一块数据后,相关的内存就可以被释放,只有当前正在处理的数据块会驻留在内存中。这是通过在回调函数中逐块导出数据而实现的。这意味着即使是非常大的数据集,也能被逐块处理,而不会导致内存溢出。
  2. Excel 导出流程
    在 SignContractExport 类中:

collection() 方法利用 getDataChunked 方法分块查询数据。
每获取一块数据(通过 map() 方法处理格式化后),立即 push 到 data 集合中,直到所有块的数据都被处理完。
最终的 Excel 导出:

在 UserController 中,当 is_down 参数为 true 时,调用 Excel::download,传入 SignContractExport 对象和相关参数。
Laravel Excel 包会利用 collection() 返回的分块数据生成 Excel 文件并下载,过程中每一块数据都是在内存中处理完成后再处理下一块,避免一次性加载所有数据导致内存溢出。
4. 内存优化效果
通过这种分块处理,大数据导出的内存占用相对于一次性加载所有数据来说大大减少。每次只需要处理一块数据,内存中只会驻留当前块的数据,而不需要一次性加载整个数据集。

这种实现方式特别适合在数据量特别大的情况下使用,如数百万行的数据导出。分块查询和处理的方式使得导出过程既高效又能避免内存耗尽的问题。

这篇关于laravel 大数据分块导出,避免内存溢出的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1090637

相关文章

Linux下利用select实现串口数据读取过程

《Linux下利用select实现串口数据读取过程》文章介绍Linux中使用select、poll或epoll实现串口数据读取,通过I/O多路复用机制在数据到达时触发读取,避免持续轮询,示例代码展示设... 目录示例代码(使用select实现)代码解释总结在 linux 系统里,我们可以借助 select、

使用EasyPoi快速导出Word文档功能的实现步骤

《使用EasyPoi快速导出Word文档功能的实现步骤》EasyPoi是一个基于ApachePOI的开源Java工具库,旨在简化Excel和Word文档的操作,本文将详细介绍如何使用EasyPoi快速... 目录一、准备工作1、引入依赖二、准备好一个word模版文件三、编写导出方法的工具类四、在Export

Python内存管理机制之垃圾回收与引用计数操作全过程

《Python内存管理机制之垃圾回收与引用计数操作全过程》SQLAlchemy是Python中最流行的ORM(对象关系映射)框架之一,它提供了高效且灵活的数据库操作方式,本文将介绍如何使用SQLAlc... 目录安装核心概念连接数据库定义数据模型创建数据库表基本CRUD操作创建数据读取数据更新数据删除数据查

前端导出Excel文件出现乱码或文件损坏问题的解决办法

《前端导出Excel文件出现乱码或文件损坏问题的解决办法》在现代网页应用程序中,前端有时需要与后端进行数据交互,包括下载文件,:本文主要介绍前端导出Excel文件出现乱码或文件损坏问题的解决办法,... 目录1. 检查后端返回的数据格式2. 前端正确处理二进制数据方案 1:直接下载(推荐)方案 2:手动构造

Python异常处理之避免try-except滥用的3个核心原则

《Python异常处理之避免try-except滥用的3个核心原则》在Python开发中,异常处理是保证程序健壮性的关键机制,本文结合真实案例与Python核心机制,提炼出避免异常滥用的三大原则,有需... 目录一、精准打击:只捕获可预见的异常类型1.1 通用异常捕获的陷阱1.2 精准捕获的实践方案1.3

C#使用iText获取PDF的trailer数据的代码示例

《C#使用iText获取PDF的trailer数据的代码示例》开发程序debug的时候,看到了PDF有个trailer数据,挺有意思,于是考虑用代码把它读出来,那么就用到我们常用的iText框架了,所... 目录引言iText 核心概念C# 代码示例步骤 1: 确保已安装 iText步骤 2: C# 代码程

Pandas处理缺失数据的方式汇总

《Pandas处理缺失数据的方式汇总》许多教程中的数据与现实世界中的数据有很大不同,现实世界中的数据很少是干净且同质的,本文我们将讨论处理缺失数据的一些常规注意事项,了解Pandas如何表示缺失数据,... 目录缺失数据约定的权衡Pandas 中的缺失数据None 作为哨兵值NaN:缺失的数值数据Panda

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别

python库pydantic数据验证和设置管理库的用途

《python库pydantic数据验证和设置管理库的用途》pydantic是一个用于数据验证和设置管理的Python库,它主要利用Python类型注解来定义数据模型的结构和验证规则,本文给大家介绍p... 目录主要特点和用途:Field数值验证参数总结pydantic 是一个让你能够 confidentl

JAVA实现亿级千万级数据顺序导出的示例代码

《JAVA实现亿级千万级数据顺序导出的示例代码》本文主要介绍了JAVA实现亿级千万级数据顺序导出的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 前提:主要考虑控制内存占用空间,避免出现同时导出,导致主程序OOM问题。实现思路:A.启用线程池