您的位置:首页 >按日期分组计算每日收支余额方法
发布于2026-04-20 阅读(0)
扫一扫,手机访问
本文详解如何基于已按日期分组的 Transaction 列表,使用 Java Stream 精确汇总每组中的收入与支出金额,并计算净余额,避免手动循环累加错误。
本文详解如何基于已按日期分组的 Transaction 列表,使用 Java Stream 精确汇总每组中的收入与支出金额,并计算净余额,避免手动循环累加错误。
在构建个人财务类应用时,一个常见需求是:将用户所有交易按日期聚合后,为每个日期动态计算「当日总收入 − 当日总支出」所得的净余额(即“剩余金额”)。您已在控制器中完成了按日期分组(TransactionGroup),但手动遍历求和易出错、逻辑冗余且难以维护。下面提供一套简洁、健壮、可复用的解决方案。
关键不在于“逐个判断再累加”,而在于利用 Stream API 对同一日期内的交易进行类型筛选与数值聚合。假设当前处理的是某个 TransactionGroup transGroup,其包含某一天的所有交易记录,推荐采用如下三步法:
LocalDate date = transGroup.getDate();
List<Transaction> transactions = transGroup.getTransactions();
// 1️⃣ 汇总当日所有 income 类型交易的 amount(自动跳过 null)
double totalIncome = transactions.stream()
.filter(t -> "INCOME".equalsIgnoreCase(t.getTransactionType().name())) // 推荐用 enum name() 而非 getDisplayName()
.mapToDouble(Transaction::getAmount)
.sum();
// 2️⃣ 汇总当日所有 expense 类型交易的 amount
double totalExpense = transactions.stream()
.filter(t -> "EXPENSE".equalsIgnoreCase(t.getTransactionType().name()))
.mapToDouble(Transaction::getAmount)
.sum();
// 3️⃣ 计算净余额(可正可负)
double dailyBalance = totalIncome - totalExpense;
System.out.printf("? %s | ? 收入: %.2f | ? 支出: %.2f | ⚖️ 结余: %.2f%n",
date, totalIncome, totalExpense, dailyBalance);? 为什么推荐 t.getTransactionType().name()?
您的 TransactionType 是 @Enumerated(EnumType.STRING),数据库存储的是 "INCOME"/"EXPENSE" 字符串。直接比对 name() 更可靠、性能更高,且避免因 getDisplayName() 实现变更(如返回中文)导致逻辑断裂。
将上述逻辑嵌入您的 getUserTransactions 方法中,为每个 TransactionGroup 添加 dailyBalance 属性,便于前端展示:
// 在循环处理 transactionByDate 前,先扩展 TransactionGroup 类(或使用 DTO)
public class TransactionGroup {
private LocalDate date;
private List<Transaction> transactions;
private double dailyBalance; // 新增字段
// ... getter/setter
public void calculateDailyBalance() {
double income = this.transactions.stream()
.filter(t -> "INCOME".equalsIgnoreCase(t.getTransactionType().name()))
.mapToDouble(Transaction::getAmount)
.sum();
double expense = this.transactions.stream()
.filter(t -> "EXPENSE".equalsIgnoreCase(t.getTransactionType().name()))
.mapToDouble(Transaction::getAmount)
.sum();
this.dailyBalance = income - expense;
}
}然后在控制器中调用:
for (Transaction t : transactions) {
if (!currDate.isEqual(t.getDate())) {
transGroup.setDate(currDate);
transGroup.setTransactions(transOnSingleDate);
transGroup.calculateDailyBalance(); // ? 关键:自动计算结余
transactionByDate.add(transGroup);
transGroup = new TransactionGroup();
transOnSingleDate = new ArrayList<>();
}
transOnSingleDate.add(t);
currDate = t.getDate();
}
// 处理最后一组
transGroup.setDate(currDate);
transGroup.setTransactions(transOnSingleDate);
transGroup.calculateDailyBalance(); // ? 不要遗漏!
transactionByDate.add(transGroup);最后,在 Thymeleaf 模板中即可直接使用:
<div th:each="group : ${transactionGroup}">
<h3 th:text="|? ${#temporals.format(group.date, 'dd/MM/yyyy')} (结余: $${group.dailyBalance})|"></h3>
<ul>
<li th:each="t : ${group.transactions}"
th:text="|• ${t.transactionType} — $${t.amount} ${t.note}|"></li>
</ul>
</div>BigDecimal income = transactions.stream()
.filter(t -> "INCOME".equals(t.getTransactionType().name()))
.map(t -> Optional.ofNullable(t.getAmount()).map(BigDecimal::valueOf).orElse(BigDecimal.ZERO))
.reduce(BigDecimal.ZERO, BigDecimal::add);通过以上重构,您不仅解决了“如何正确求和并相减”的技术问题,更建立了清晰、可测试、易扩展的财务计算模型——这是构建可信财务功能的坚实基础。
上一篇:PHP框架搭建优势解析
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
正版软件
正版软件
正版软件
正版软件
1
2
3
7
9