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

相关文章

Redis的数据过期策略和数据淘汰策略

《Redis的数据过期策略和数据淘汰策略》本文主要介绍了Redis的数据过期策略和数据淘汰策略,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录一、数据过期策略1、惰性删除2、定期删除二、数据淘汰策略1、数据淘汰策略概念2、8种数据淘汰策略

轻松上手MYSQL之JSON函数实现高效数据查询与操作

《轻松上手MYSQL之JSON函数实现高效数据查询与操作》:本文主要介绍轻松上手MYSQL之JSON函数实现高效数据查询与操作的相关资料,MySQL提供了多个JSON函数,用于处理和查询JSON数... 目录一、jsON_EXTRACT 提取指定数据二、JSON_UNQUOTE 取消双引号三、JSON_KE

Python给Excel写入数据的四种方法小结

《Python给Excel写入数据的四种方法小结》本文主要介绍了Python给Excel写入数据的四种方法小结,包含openpyxl库、xlsxwriter库、pandas库和win32com库,具有... 目录1. 使用 openpyxl 库2. 使用 xlsxwriter 库3. 使用 pandas 库

SpringBoot定制JSON响应数据的实现

《SpringBoot定制JSON响应数据的实现》本文主要介绍了SpringBoot定制JSON响应数据的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们... 目录前言一、如何使用@jsonView这个注解?二、应用场景三、实战案例注解方式编程方式总结 前言

使用Python在Excel中创建和取消数据分组

《使用Python在Excel中创建和取消数据分组》Excel中的分组是一种通过添加层级结构将相邻行或列组织在一起的功能,当分组完成后,用户可以通过折叠或展开数据组来简化数据视图,这篇博客将介绍如何使... 目录引言使用工具python在Excel中创建行和列分组Python在Excel中创建嵌套分组Pyt

在Rust中要用Struct和Enum组织数据的原因解析

《在Rust中要用Struct和Enum组织数据的原因解析》在Rust中,Struct和Enum是组织数据的核心工具,Struct用于将相关字段封装为单一实体,便于管理和扩展,Enum用于明确定义所有... 目录为什么在Rust中要用Struct和Enum组织数据?一、使用struct组织数据:将相关字段绑

在Mysql环境下对数据进行增删改查的操作方法

《在Mysql环境下对数据进行增删改查的操作方法》本文介绍了在MySQL环境下对数据进行增删改查的基本操作,包括插入数据、修改数据、删除数据、数据查询(基本查询、连接查询、聚合函数查询、子查询)等,并... 目录一、插入数据:二、修改数据:三、删除数据:1、delete from 表名;2、truncate

Java导出Excel动态表头的示例详解

《Java导出Excel动态表头的示例详解》这篇文章主要为大家详细介绍了Java导出Excel动态表头的相关知识,文中的示例代码简洁易懂,具有一定的借鉴价值,有需要的小伙伴可以了解下... 目录前言一、效果展示二、代码实现1.固定头实体类2.动态头实现3.导出动态头前言本文只记录大致思路以及做法,代码不进

C#多线程编程中导致死锁的常见陷阱和避免方法

《C#多线程编程中导致死锁的常见陷阱和避免方法》在C#多线程编程中,死锁(Deadlock)是一种常见的、令人头疼的错误,死锁通常发生在多个线程试图获取多个资源的锁时,导致相互等待对方释放资源,最终形... 目录引言1. 什么是死锁?死锁的典型条件:2. 导致死锁的常见原因2.1 锁的顺序问题错误示例:不同

Java实现Elasticsearch查询当前索引全部数据的完整代码

《Java实现Elasticsearch查询当前索引全部数据的完整代码》:本文主要介绍如何在Java中实现查询Elasticsearch索引中指定条件下的全部数据,通过设置滚动查询参数(scrol... 目录需求背景通常情况Java 实现查询 Elasticsearch 全部数据写在最后需求背景通常情况下