Rob*_*ain 2 java lambda java-8 java-stream collectors
我正在尝试使用Java-8 lambdas来解决以下问题:
给定一个List<Transaction>
,对于每一个Category.minorCategory
我需要的总和Transaction.amount
每Category.minorCategory
和Map
的Transaction.accountNumber
与总和Transaction.amount
每Transaction.accountNumber
.根据下面的代码,我有这个工作.我现在要求分组Category.majorCategory
,基本上返回一个Map<String, Map<String, MinorCategorySummary>>
键入Category.majorCategory
.
我一直在努力,直到分组的阶段,Category.majorCategory
但很难看到解决方案; 编程与lambdas的范式转换证明了一个陡峭的学习曲线.
TransactionBreakdown
是行动发生的地方,我想回到哪里Map<String, Map<String, MinorCategorySummary>>
.
public class Transaction {
private final String accountNumber;
private final BigDecimal amount;
private final Category category;
}
public class Category {
private final String majorCategory;
private final String minorCategory;
}
public class MinorCategorySummary {
private final BigDecimal sumOfAmountPerMinorCategory;
private final Map<String, BigDecimal> accountNumberSumOfAmountMap;
private final Category category;
}
public class TransactionBreakdown {
Function<Entry<String, List<Transaction>>, MinorCategorySummary> transactionSummariser = new TransactionSummariser();
public Map<Object, MinorCategorySummary> getTransactionSummaries(List<Transaction> transactions) {
return transactions
.stream()
.collect(groupingBy(t -> t.getCategory().getMinorCategory()))
.entrySet()
.stream()
.collect(
toMap(Entry::getKey,
transactionSummariser));
}
}
public class TransactionSummariser implements Function<Entry<String, List<Transaction>>, MinorCategorySummary> {
@Override
public MinorCategorySummary apply(
Entry<String, List<Transaction>> entry) {
return new MinorCategorySummary(
entry.getValue()
.stream()
.map(Transaction::getAmount)
.collect(reducing(BigDecimal.ZERO, BigDecimal::add)),
entry.getValue()
.stream()
.collect(
groupingBy(Transaction::getAccountNumber,
mapping(Transaction::getAmount,
reducing(BigDecimal.ZERO, BigDecimal::add)))),
entry.getValue().get(0).getCategory());
}
}
Run Code Online (Sandbox Code Playgroud)
你的课堂设计对我来说很奇怪.为什么将类别放入摘要类只是为了将类别作为映射键?有一个没有类别的摘要类会更有意义:
public class TransactionSummary {
private final BigDecimal amount;
private final Map<String, BigDecimal> acctToTotal;
TransactionSummary(Map<String, BigDecimal> acctToTotal) {
this.acctToTotal = Collections.unmodifiableMap(acctToTotal);
this.amount = acctToTotal.values().stream()
.reduce(BigDecimal.ZERO, BigDecimal::add);
}
public static Collector<Transaction, ?, TransactionSummary> collector() {
// this can be a static constant
return collectingAndThen(
toMap(Transaction::getAccountNumber,Transaction::getAmount,BigDecimal::add),
TransactionSummary::new
);
}
// getters
}
Run Code Online (Sandbox Code Playgroud)
现在你的两个问题已经清楚地解决了,没有冗余:
Map<String, TransactionSummary> minorSummary = transactions.stream()
.collect(groupingBy(
t -> t.getCategory().getMinorCategory(),
TransactionSummary.collector()
));
Map<String, Map<String, TransactionSummary>> majorMinorSummary = transactions.stream()
.collect(groupingBy(
t -> t.getCategory().getMajorCategory(),
groupingBy(
t -> t.getCategory().getMinorCategory(),
TransactionSummary.collector()
)
));
Run Code Online (Sandbox Code Playgroud)