0

0

Laravel Collection多级分组与聚合求和教程

花韻仙語

花韻仙語

发布时间:2025-09-21 12:35:38

|

661人浏览过

|

来源于php中文网

原创

Laravel Collection多级分组与聚合求和教程

本教程详细阐述了如何利用Laravel Collection的强大功能,对复杂嵌套数据进行多级分组,并在此基础上对特定字段进行聚合求和。通过实例演示,我们将学习如何处理groupBy操作后产生的多层嵌套结构,并通过链式map方法深入到最内层数据,实现精确的数据汇总与结构重塑,最终输出简洁明了的聚合结果。

1. 引言:数据聚合的常见需求

在实际应用开发中,我们经常需要对数据进行统计和汇总。例如,在一个销售订单系统中,可能存在多条相同商品但来自不同批次或不同规格的记录,我们希望将这些记录合并,并计算出它们的总数量。laravel collection提供了强大的工具来处理这类复杂的数据转换和聚合任务。

考虑一个场景:我们有一组待发货的圣诞树记录,每棵树都有type(种类)、size(尺寸)、amount(数量)等属性,并且可能包含slot、pallet、label等批次或包装信息。当同一类型和尺寸的树木在多批次中出现时,我们希望将它们合并为一条记录,并汇总其总数量,同时移除不必要的批次细节信息。

原始数据结构示例(line_items):

[
  {
    "slot": 2,
    "pallet": "cghjh",
    "type": "NGR",
    "label": "purple",
    "size": "125-150",
    "amount": "30"
  },
  {
    "slot": 3,
    "pallet": "cghjh",
    "type": "NGR",
    "label": "purple",
    "size": "125-150",
    "amount": "30"
  },
  {
    "slot": 2,
    "pallet": "yghiuj",
    "type": "NGR",
    "label": "orange",
    "size": "150-175",
    "amount": "30"
  },
  {
    "slot": 3,
    "pallet": "cghjh",
    "type": "NOB",
    "label": "purple",
    "size": "125-150",
    "amount": "30"
  }
]

我们的目标是将其转换为以下聚合后的结构:

{
  "NGR": {
    "125-150": [
      {
        "type": "NGR",
        "size": "125-150",
        "amount": "60"
      }
    ],
    "150-175": [
      {
        "type": "NGR",
        "size": "150-175",
        "amount": "30"
      }
    ]
  },
  "NOB": {
    "125-150": [
      {
        "type": "NOB",
        "size": "125-150",
        "amount": "30"
      }
    ]
  }
}

2. 初步分组:使用 groupBy

Laravel Collection的 groupBy 方法是实现数据聚合的第一步。它可以根据一个或多个键将集合中的元素分组。当提供一个键数组时,groupBy会创建嵌套的分组结构。

use Illuminate\Support\Collection;

// 假设 $deliveryNote->line_items 是上述的原始数据数组
$lineItems = collect([
    // ... 原始数据示例 ...
    [ "slot" => 2, "pallet" => "cghjh", "type" => "NGR", "label" => "purple", "size" => "125-150", "amount" => "30" ],
    [ "slot" => 3, "pallet" => "cghjh", "type" => "NGR", "label" => "purple", "size" => "125-150", "amount" => "30" ],
    [ "slot" => 2, "pallet" => "yghiuj", "type" => "NGR", "label" => "orange", "size" => "150-175", "amount" => "30" ],
    [ "slot" => 3, "pallet" => "cghjh", "type" => "NOB", "label" => "purple", "size" => "125-150", "amount" => "30" ]
]);

$groupedData = $lineItems->groupBy(['type', 'size']);

执行上述 groupBy(['type', 'size']) 后,$groupedData 的结构将如下所示:

{
  "NGR": {
    "125-150": [
      { /* 原始NGR, 125-150的第一个对象 */ },
      { /* 原始NGR, 125-150的第二个对象 */ }
    ],
    "150-175": [
      { /* 原始NGR, 150-175的对象 */ }
    ]
  },
  "NOB": {
    "125-150": [
      { /* 原始NOB, 125-150的对象 */ }
    ]
  }
}

可以看到,数据已经按照 type 和 size 进行了两级分组,但此时 amount 尚未求和,且每个分组内仍包含原始的详细信息。

E酷购网络商城建站程序
E酷购网络商城建站程序

一套傻瓜式的建站程序,由前台购物、后台管理、在线支付三部分组成介绍说明:1.注册与否均可购物(同类程序大多要求注册才能购物),方便了那些懒得注册的客户。降低用户使用门槛,自然可抓住更多潜在商机。2.会员等级和折扣功能。管理员可方便的为会员设置不同等级,不同等级的员会可享受不同的购物折扣。3.站内短信、留言发布,沟通无极限。会员和游客均可发送短信和留言。4.完美融合在线支付功能,无需编程、无需修改源

下载

3. 深入聚合:使用嵌套 map 进行求和与结构重塑

简单的在 groupBy 之后直接使用 map 和 sum 并不能达到预期效果,因为 groupBy(['type', 'size']) 产生的是一个嵌套的集合结构。我们需要进行两层 map 操作来遍历这个结构并进行聚合。

核心思路:

  1. 外层 map 遍历第一级分组(例如 type,如 "NGR", "NOB")。
  2. 内层 map 遍历第二级分组(例如 size,如 "125-150", "150-175")。
  3. 在最内层的 map 中,对相同 type 和 size 的所有项进行 amount 求和,并提取 type 和 size 字段以构建新的精简对象。
$aggregatedData = $groupedData->map(function ($itemsGroupedBySize) {
    // $itemsGroupedBySize 是一个Collection,其键是 'size' (e.g., "125-150")
    // 对应的值是另一个Collection,包含该 type 和 size 的所有原始 line_items。
    return $itemsGroupedBySize->map(function ($individualItems) {
        // $individualItems 是一个Collection,包含所有具有相同 type 和 size 的原始项目。
        // 例如:[{"slot": 2, "type": "NGR", "size": "125-150", "amount": "30"}, {"slot": 3, "type": "NGR", "size": "125-150", "amount": "30"}]

        // 从分组中的任意一个元素(例如第一个)获取 type 和 size
        $firstItem = $individualItems->first();

        return [
            'type' => $firstItem->type,
            'size' => $firstItem->size,
            // 对当前分组中的所有 'amount' 进行求和
            'amount' => $individualItems->sum('amount'),
        ];
    });
});

上述代码执行后,$aggregatedData 将得到我们期望的聚合结果:

{
  "NGR": {
    "125-150": [
      {
        "type": "NGR",
        "size": "125-150",
        "amount": 60 // 注意:如果原始amount是字符串,sum()会自动尝试转换为数字
      }
    ],
    "150-175": [
      {
        "type": "NGR",
        "size": "150-175",
        "amount": 30
      }
    ]
  },
  "NOB": {
    "125-150": [
      {
        "type": "NOB",
        "size": "125-150",
        "amount": 30
      }
    ]
  }
}

4. 注意事项与最佳实践

  • 数据类型转换: Collection::sum() 方法在遇到字符串类型的数字时,会尝试将其转换为数字进行求和。如果你的 amount 字段可能包含非数字字符,建议在求和前进行显式转换或验证。
  • first() 的使用: 在内层 map 中,我们使用 ->first() 来获取 type 和 size。这假定每个分组($individualItems)至少包含一个元素,这在 groupBy 操作后是必然成立的。
  • 链式操作的可读性: 尽管可以将所有操作链式写在一起,但为了代码可读性,特别是对于复杂的嵌套聚合,将其分解为多个步骤(如先 groupBy 再 map)是一个好习惯。
  • 性能考量: 对于非常大的数据集,多次 map 操作可能会有性能开销。在极端情况下,可以考虑使用数据库层面的聚合查询来优化性能。然而,对于大多数Web应用场景,Laravel Collection的性能已足够优秀。
  • 灵活调整输出结构: 如果最终需要的输出结构与上述示例略有不同,你可以在内层 map 的返回数组中灵活调整键值对,以匹配你的需求。例如,如果你不希望最内层是一个数组,而是直接一个对象,可以调整 return 语句。

5. 总结

Laravel Collection为PHP开发者提供了极其灵活和强大的数据处理能力。通过理解 groupBy 如何创建嵌套结构,并结合嵌套的 map 操作,我们可以有效地对复杂数据进行多级分组、聚合求和以及结构重塑。这种模式不仅限于简单的求和,还可以扩展到平均值、最大值、最小值等其他聚合函数,极大地简化了数据处理的复杂性,提高了开发效率。掌握这些技巧,将使你在处理各种数据转换任务时游刃有余。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
laravel组件介绍
laravel组件介绍

laravel 提供了丰富的组件,包括身份验证、模板引擎、缓存、命令行工具、数据库交互、对象关系映射器、事件处理、文件操作、电子邮件发送、队列管理和数据验证。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

320

2024.04.09

laravel中间件介绍
laravel中间件介绍

laravel 中间件分为五种类型:全局、路由、组、终止和自定。想了解更多laravel中间件的相关内容,可以阅读本专题下面的文章。

278

2024.04.09

laravel使用的设计模式有哪些
laravel使用的设计模式有哪些

laravel使用的设计模式有:1、单例模式;2、工厂方法模式;3、建造者模式;4、适配器模式;5、装饰器模式;6、策略模式;7、观察者模式。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

373

2024.04.09

thinkphp和laravel哪个简单
thinkphp和laravel哪个简单

对于初学者来说,laravel 的入门门槛较低,更易上手,原因包括:1. 更简单的安装和配置;2. 丰富的文档和社区支持;3. 简洁易懂的语法和 api;4. 平缓的学习曲线。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

374

2024.04.10

laravel入门教程
laravel入门教程

本专题整合了laravel入门教程,想了解更多详细内容,请阅读专题下面的文章。

86

2025.08.05

laravel实战教程
laravel实战教程

本专题整合了laravel实战教程,阅读专题下面的文章了解更多详细内容。

69

2025.08.05

laravel面试题
laravel面试题

本专题整合了laravel面试题相关内容,阅读专题下面的文章了解更多详细内容。

68

2025.08.05

数据类型有哪几种
数据类型有哪几种

数据类型有整型、浮点型、字符型、字符串型、布尔型、数组、结构体和枚举等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

310

2023.10.31

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

14

2026.01.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 10.4万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 11.2万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号