当前位置: 首页 > news >正文

JavaScript 根据数组对象的某个 指定属性 分组合并,并累加金额

/** * 数组对象按指定属性分组,合并金额 * @param {Array} arr - 原始数组(元素为对象) * @param {string} groupKey - 分组依据的属性名(如 'type'、'id') * @param {string} moneyKey - 需合并的金额属性名(如 'money'、'amount') * @returns {Array} 合并后的数组 */functionmergeObjectsByKey(arr,groupKey,moneyKey){// 用 Map 分组(比普通对象更安全,支持任意类型键)constresultMap=newMap();arr.forEach(item=>{// 1. 获取当前 item 的分组键值(如 item.type = 'A')constkey=item[groupKey];// 2. 确保金额为数字(兼容字符串格式的金额,如 '100' → 100)constcurrentMoney=Number(item[moneyKey])||0;// 3. 检查 Map 中是否已存在该分组if(resultMap.has(key)){// 存在:累加金额constexistingItem=resultMap.get(key);existingItem[moneyKey]+=currentMoney;}else{// 不存在:创建新分组对象(深拷贝避免修改原数据)constnewItem={...item};// 复制原对象所有属性newItem[moneyKey]=currentMoney;// 初始化金额(确保为数字)resultMap.set(key,newItem);}});// 4. 将 Map 的值转为数组返回returnArray.from(resultMap.values());}

场景:按 type 分组,合并 money 金额

// 原始数组constoriginalArr=[{id:1,type:'水果',name:'苹果',money:10},{id:2,type:'蔬菜',name:'白菜',money:5},{id:3,type:'水果',name:'香蕉',money:15},// 同 type='水果'{id:4,type:'肉类',name:'猪肉',money:30},{id:5,type:'蔬菜',name:'萝卜',money:'8'},// 金额为字符串格式];// 调用函数:按 type 分组,合并 moneyconstmergedArr=mergeObjectsByKey(originalArr,'type','money');console.log(mergedArr);

输出结果(按 type 分组,金额累加):

[{id:1,type:'水果',name:'苹果',money:25},// 10 + 15 = 25{id:2,type:'蔬菜',name:'白菜',money:13},// 5 + 8 = 13(字符串自动转数字){id:4,type:'肉类',name:'猪肉',money:30}]

关键说明
分组逻辑:用 Map 存储分组,键为 groupKey 的值(如 ‘水果’),值为合并后的对象。Map 比普通对象更安全(避免键名冲突,支持数字 / 对象等类型的键)。
金额兼容:通过 Number(item[moneyKey]) || 0 处理:
数字金额(如 10)直接使用;
字符串金额(如 ‘8’)自动转为数字;
缺失金额或非数字(如 undefined、‘abc’)视为 0。
属性保留:用 { …item } 浅拷贝原对象的所有属性(如 id、name),合并时仅更新金额,其他属性保留第一个分组对象的值(若需保留所有属性,可修改为数组存储,如 name: [‘苹果’, ‘香蕉’])。
扩展需求:保留分组内所有非金额属性
如果需要保留同组所有对象的非金额属性(如合并 name 为数组),可修改函数:

functionmergeObjectsWithAllProps(arr,groupKey,moneyKey){constresultMap=newMap();arr.forEach(item=>{constkey=item[groupKey];constcurrentMoney=Number(item[moneyKey])||0;if(resultMap.has(key)){constexisting=resultMap.get(key);existing[moneyKey]+=currentMoney;// 非金额属性转为数组(去重)Object.keys(item).forEach(prop=>{if(prop!==groupKey&&prop!==moneyKey){if(!existing[prop].includes(item[prop])){existing[prop].push(item[prop]);}}});}else{constnewItem={[moneyKey]:currentMoney};// 分组键保留原值,其他属性转为数组Object.keys(item).forEach(prop=>{newItem[prop]=prop===groupKey?item[prop]:[item[prop]];});resultMap.set(key,newItem);}});returnArray.from(resultMap.values());}

扩展示例输出:

[{type:'水果',id:[1,3],name:['苹果','香蕉'],money:25},// ... 其他分组]``` 边界情况处理 原始数组为空:返回空数组; 部分对象缺失 groupKey:按undefined分组; 金额为0或负数:正常累加(符合业务逻辑); 分组键为复杂类型(如对象):Map 可直接支持(普通对象会转为字符串[object Object],不推荐用对象作为分组键)。
http://www.cnnetsun.cn/news/64822.html

相关文章:

  • 通过SEO推广LobeChat博客内容,带动大模型Token购买转化
  • 【Svelte】重定向页面
  • 基于SpringBoot的日用品仓储管理系统的设计与实现
  • 基于SpringBoot的校园论坛交流系统
  • AutoGPT如何处理模糊目标?自然语言理解边界探讨
  • 清华镜像站推荐:Miniconda下载提速80%的秘密武器
  • update.py update脚本 git一键上传push脚本 - Git自动化推送代码的几种方式及实用脚本
  • 从GitHub获取Qwen3-8B最新镜像并完成本地化部署
  • Ubuntu安装完成后配置PyTorch-GPU的完整流程
  • 购买GPU算力租用Qwen3-14B实例的性价比分析
  • LobeChat前端性能优化建议:减少加载时间提升访问量
  • 学术研究新利器:Qwen3-8B开箱即用镜像发布
  • 使用wget命令从清华源下载PyTorch安装包的脚本示例
  • AutoGPT镜像适用于科研场景吗?高校团队已投入使用
  • GitHub热门项目推荐:基于Qwen3-14B开发的企业级AI助手
  • 从零到网络安全专家:一张全景路线图(2025版)
  • LeetCode 46/51 排列型回溯题笔记-全排列 / N 皇后
  • 一周回顾:勒索飙升、AI上阵、人形机器人被盯上
  • 嵌入式FOTA进阶:文件系统直接升级+串口分段传输深度指南!
  • AutoGPT提示词工程技巧:提升任务拆解准确性
  • Stable Diffusion AIGC 视觉设计实战教程之 07-图生图
  • 当毕业论文不再是“一个人的深夜战场”:一位研究生眼中的AI科研协作者如何重塑写作流程
  • 统计提交svn代码行数,文件以及文档
  • 解锁学术新次元:书匠策AI科研工具为毕业论文注入智慧动能
  • GPT-5.2全面解析:AI“打工能力“大提升,程序员职场必备技能
  • vue-springboot基于Java医院药品管理系统的设计与实现_8z88u88g
  • 深圳 CNC 加工哪家强?慧闻智造!精密零件加工的靠谱专家
  • Java中高级面试题详解(十五):彻底搞懂 Spring Boot 启动流程与扩展点,别再只会写 main 方法!
  • CTF 解题核心思维 + 新手入门全攻略
  • 2026PCB产业高端化浪潮与慕尼黑上海电子展的连接枢纽