游乐游手机版
首页/前端开发/文章详情

vxe-table数据分组统计与表尾合计实现方法

时间:2026-06-13 06:50
在数据报表或管理系统里,按某个字段分组并计算每组统计值(比如求和、平均值),同时表格底部还要显示所有数据的合计——这个需求太常见了。vxe-table 正好提供了灵活的组合方案:aggregateConfig 配上自定义计算函数 calcValuesMethod,再手动维护一个表尾 footerDa

在数据报表或管理系统里,按某个字段分组并计算每组统计值(比如求和、平均值),同时表格底部还要显示所有数据的合计——这个需求太常见了。vxe-table 正好提供了灵活的组合方案:aggregateConfig 配上自定义计算函数 calcValuesMethod,再手动维护一个表尾 footerData,就能轻松搞定。

下面拿一个完整的例子来说明:按角色(role)对数据分组,统计每个角色的 num 总和,最后在表格底部显示所有数据的 num 总计。

属性说明

配置项 / 属性所在位置作用
aggregateConfig.groupFields表格配置(gridOptions)指定分组字段,例如 ['role'],表格会按该字段对数据进行分组,并在每个分组后插入“分组合计行”
aggregateConfig.calcValuesMethod表格配置自定义分组统计值计算函数,接收当前分组的子数据(children)和列信息(column),返回统计值
aggFunc列配置(columns)标记该列需要参与聚合计算,配合分组统计使用
footerData表格配置手动指定表尾合计行的数据,适用于简单的总计、平均值等,更复杂时可配合 updateFooter 方法动态计算
rowGroupNode列配置标识该列作为分组的显示节点,通常设置为 true,使分组后的行可以展开/折叠(示例中使用)

注意:分组聚合需要同时设置 aggregateConfig.groupFields 和至少一列的 rowGroupNode: true,同时要聚合的列必须设置 aggFunc: true(或自定义聚合函数)。

代码

<template>
  <div>
    <vxe-grid v-bind="gridOptions">vxe-grid>
  div>
template><script setup>
import { reactive } from 'vue'// 表尾合计行数据(手动维护总计)
const footerSummaryRow = reactive({
  seq: '总计',
  num: 0
})const gridOptions = reactive({
  border: true,
  showOverflow: true,       // 超出显示省略号
  showFooter: true,         // 开启表尾
  height: 500,
  loading: false,  // 分组聚合配置
  aggregateConfig: {
    groupFields: ['role'],   // 按角色字段分组
    // 自定义分组统计值的计算方法
    calcValuesMethod({ column, children }) {
      // 只对 num 列进行求和
      if (column.field === 'num') {
        let sum = 0
        children.forEach(item => {
          sum += item.num
        })
        return sum
      }
      // 其他列不统计,返回 0 或空字符串
      return 0
    }
  },  // 列定义
  columns: [
    { type: 'seq', width: 70, title: '序号' },
    { field: 'name', title: '姓名', minWidth: 150, rowGroupNode: true },  // 分组显示节点
    { field: 'role', title: '角色', width: 120 },
    { field: 'num', title: '数量', width: 100, aggFunc: true },           // 参与聚合
    { field: 'age', title: '年龄', width: 80 },
    { field: 'address', title: '地址', minWidth: 200 }
  ],  data: [],               // 动态加载的数据
  footerData: [footerSummaryRow]   // 表尾数据
})// 模拟异步加载数据,并计算表尾总计
const loadData = () => {
  gridOptions.loading = true
  setTimeout(() => {
    const rawData = [
      { id: 10001, name: 'Test1', role: 'Develop', sex: 'Woman', age: 28, num: 63, date: '2025-02-01', address: 'test abc' },
      { id: 10002, name: 'Test2', role: 'Test', sex: 'Man', age: 22, num: 63, date: '2025-01-01', address: 'Guangzhou' },
      { id: 10003, name: 'Test3', role: 'PM', sex: 'Woman', age: 32, num: 10, date: '2025-05-01', address: 'Shanghai' },
      { id: 10004, name: 'Test4', role: 'Designer', sex: 'Man', age: 32, num: 24, date: '2025-01-01', address: 'test abc' },
      { id: 10005, name: 'Test5', role: 'Develop', sex: 'Man', age: 30, num: 98, date: '2025-01-01', address: 'Shanghai' },
      { id: 10006, name: 'Test6', role: 'Designer', sex: 'Man', age: 30, num: 46, date: '2025-03-01', address: 'test abc' },
      { id: 10007, name: 'Test7', role: 'Test', sex: 'Woman', age: 29, num: 35, date: '2025-05-01', address: 'test abc' },
      { id: 10008, name: 'Test8', role: 'PM', sex: 'Woman', age: 35, num: 84, date: '2025-11-01', address: 'test abc' },
      { id: 10009, name: 'Test9', role: 'Test', sex: 'Man', age: 21, num: 25, date: '2025-05-01', address: 'test abc' },
      { id: 10010, name: 'Test10', role: 'PM', sex: 'Woman', age: 28, num: 32, date: '2025-03-01', address: 'test abc' },
      { id: 10011, name: 'Test11', role: 'Test', sex: 'Woman', age: 29, num: 24, date: '2025-03-01', address: 'test abc' },
      { id: 10012, name: 'Test12', role: 'Develop', sex: 'Man', age: 37, num: 28, date: '2025-10-01', address: 'test abc' },
      { id: 10013, name: 'Test13', role: 'Test', sex: 'Woman', age: 24, num: 38, date: '2025-02-01', address: 'test abc' },
      { id: 10014, name: 'Test14', role: 'Develop', sex: 'Man', age: 34, num: 46, date: '2025-08-01', address: 'test abc' },
      { id: 10015, name: 'Test15', role: 'Designer', sex: 'Man', age: 21, num: 48, date: '2025-05-01', address: 'test abc' },
      { id: 10016, name: 'Test16', role: 'Designer', sex: 'Woman', age: 21, num: 59, date: '2025-10-01', address: 'test abc' },
      { id: 10017, name: 'Test17', role: 'Test', sex: 'Man', age: 31, num: 81, date: '2025-12-01', address: 'test abc' },
      { id: 10018, name: 'Test18', role: 'Develop', sex: 'Woman', age: 32, num: 75, date: '2025-10-01', address: 'test abc' },
      { id: 10019, name: 'Test19', role: 'Test', sex: 'Man', age: 37, num: 80, date: '2025-02-01', address: 'test abc' },
      { id: 10020, name: 'Test20', role: 'Develop', sex: 'Man', age: 41, num: 60, date: '2025-03-01', address: 'test abc' }
    ]    // 计算总计(num 总和)
    let totalNum = 0
    rawData.forEach(row => {
      totalNum += row.num
    })
    footerSummaryRow.num = totalNum    gridOptions.data = rawData
    gridOptions.loading = false
  }, 200)
}// 调用加载
loadData()
script>

实现思路

  • 分组显示:表格会按 role 字段将数据分组,同一角色的数据连续排列,并在每个分组前显示一个可展开/折叠的分组行(分组标题行)。
  • 分组统计行:在每个分组的末尾,自动插入一行“分组合计”,其中 num 列显示该组内所有 num 值的总和(由 calcValuesMethod 自定义计算)。
  • 表尾总计:表格底部固定一行(footerData),显示所有数据的 num 总计。该总计需要手动从数据源计算并赋值。

总的来说,vxe-table 通过 aggregateConfigaggFunc,再搭配自定义计算函数 calcValuesMethod,就能轻松应对各种复杂的分组统计场景。而表尾合计通过 footerData 手动维护,既灵活又直观。把这两者结合起来,一个带分组统计和总计的专业数据表格就能快速成型。

来源:https://juejin.cn/post/7645900195423305754
上一篇OpenSpec+SDD规范驱动AI智能体开发项目深度实战全攻略 下一篇Go语言Headless后端脚手架:企业级多栈管理实践
本站内容用于信息整理与展示,如有侵权或内容问题请及时联系处理。

相关推荐

补充同频道和同主题内容,方便继续浏览更多相关内容。

同类最新

继续查看同栏目最近更新的文章。

更多
如何在JavaScript中实现基于旋转视野的FOV射线绘制详解
前端开发 · 2026-07-01

如何在JavaScript中实现基于旋转视野的FOV射线绘制详解

如果用一句话概括核心,那就是:在 RayCasting 游戏开发中,绘制动态视野边界线(FOV)最可靠的方式是在逻辑层通过数学公式将坐标“算”出来,而不是依赖 Canvas 绘图上下文的旋转操作。 在实现类似 Doom 风格的 RayCasting 游戏时,动态视野(Field of View, F

TypeScript后端数据正确映射为前端接口类型的方法
前端开发 · 2026-07-01

TypeScript后端数据正确映射为前端接口类型的方法

在后端数据与前端类型之间来回转换,几乎是每位 TypeScript 开发者都无法回避的常态。后端返回的 car_brand、reg_number,和前端接口中定义的 brand、govtNumber,命名风格常常对不上号。此时,如果为了省事直接用 as 类型断言“强行”指认类型,那就踩进了常见的陷阱

动态HTML表格按层级条件合并单元格的JavaScript实现
前端开发 · 2026-07-01

动态HTML表格按层级条件合并单元格的JavaScript实现

本文详细讲解一种递归式 JavaScript 合并单元格方法,用于按列优先级(如前3列)智能合并表格行:仅当前一列已合并的前提下,才允许后续列合并相同值,从而精准实现多级分组与层级表格合并效果。 在动态生成的 HTML 表格中,按业务逻辑合并重复行是常见需求。然而,简单地对单列分别遍历合并——例如先

Next.js 13+重定向后滚动失效解决方案
前端开发 · 2026-07-01

Next.js 13+重定向后滚动失效解决方案

在 Next js App Router 的日常开发中,有一个令人颇为困扰的异常现象——当服务端执行 `redirect()` 跳转后,目标页面竟然无法正常滚动。没错,页面已经渲染完成,内容也完整显示,但垂直滚动条仿佛凭空消失。这个问题在 Next js 13 5 4 版本中尤为突出。 先给出结论:

WebGL图像加载延迟的纹理初始化时立即显示方法
前端开发 · 2026-07-01

WebGL图像加载延迟的纹理初始化时立即显示方法

本文详细介绍如何利用 Promise 与 async await 重构 WebGL 纹理加载流程,彻底解决首次渲染显示蓝色占位色、需要手动交互才能刷新的问题,实现文件导入后四张纹理平面即时正确渲染。 实际上,这个坑在 WebGL 开发中相当常见——纹理异步加载的小陷阱,说起来不大,但第一次遇到确实令