I know there a similar questions asked in the forum but none of them seem to be addressing my problem fully. Now I'm very new to Java 8, so please bear with me. I have a list of Products, for example:
Input:
name category type cost
prod1 cat2 t1 100.23
prod2 cat1 t2 50.23
prod1 cat1 t3 200.23
prod3 cat2 t1 150.23
prod1 cat2 t1 100.23
Output:
Single line (name, category, type) summing the cost and count of products.
Product {
public String name;
public String category;
public String type;
public int id;
public double cost;
}
I need to group this by name, category and type and produce a single result that summarizes this data and produces the total cost and count of each product. Most examples show grouping by two fields and aggregating using single criteria.
Following suggestions on the forumn, i came up with this for groupings:
public class ObjectKeys {
ArrayList<Object> keys;
public ObjectKeys(Object...searchKeys) {
keys = new ArrayList<Object>();
for (int i = 0; i < searchKeys.length; i++) {
keys.add( searchKeys[i] );
}
}
}
Then used it as follows:
Map<String, Map<String, Map<String, List<Product>>>> productsByNameCategoryType =
products.stream().collect(groupingBy(new ObjectKeys(l.name(), l.category(),l.type())))
But how do I chain count and sum to the the above code? Especially for group by with more than 2 fields. Is there a better way to do this?
Like I mentioned my Java8 is not that good, please help.
class Product {
public String name;
public String category;
public String type;
public int id;
//todo:implement equals(), toString() and hashCode()
}
class Item{
public Product product;
public double cost;
}
you can summarizing items grouping by product by using Collectors#groupingBy & Collectors#summarizingDouble.
List<Item> items = ...;
Map<Product, DoubleSummaryStatistics> stat = items.stream().collect(groupingBy(
it -> it.product,
Collectors.summarizingDouble(it -> it.cost)
));
// get some product summarizing
long count = stat.get(product).getCount();
double sum = stat.get(product).getSum();
//list all product summarizing
stat.entrySet().forEach(it ->
System.out.println(String.format("%s - count: %d, total cost: %.2f"
, it.getKey(),it.getValue().getCount(), it.getValue().getSum()));
);
First, you need adding a qty
field in Item
class:
class Item{
public int qty;
//other fields will be omitted
public Item add(Item that) {
if (!Objects.equals(this.product, that.product)) {
throw new IllegalArgumentException("Can't be added items"
+" with diff products!");
}
return from(product, this.cost + that.cost, this.qty + that.qty);
}
private static Item from(Product product, double cost, int qty) {
Item it = new Item();
it.product = product;
it.cost = cost;
it.qty = qty;
return it;
}
}
then you can using Collectors#toMap to merges items with same product:
Collection<Item> summarized = items.stream().collect(Collectors.toMap(
it -> it.product,
Function.identity(),
Item::add
)).values();
you can see both ways doing the same thing, but the second approach is easier to operates on a stream. and the tests of two ways I having checked in github, you can click and see more details: summarizing items & merge items ways.
Here’s the quick and dirty solution:
Map<String, String> productsByNameCategoryType = products.stream()
.collect(Collectors.groupingBy(p
-> p.getName() + '-' + p.getCategory() + '-' + p.getType(),
Collectors.collectingAndThen(
Collectors.summarizingDouble(Product::getCost),
dss -> String.format("%7.2f%3d",
dss.getSum(), dss.getCount()))));
You will probably want to build your own classes for both the keys and the values of the result map. In any case, with your data and the above code the map contains four entries:
prod1-cat1-t3: 200,23 1
prod1-cat2-t1: 200,46 2
prod3-cat2-t1: 150,23 1
prod2-cat1-t2: 50,23 1
The sums are printed with comma as decimal point because my computer has Danish locale (you can pass a locale to String.format()
to control the locale if you want).
Your friend is the combination of Collectors.collectingAndThen()
and Collectors.summarizingDouble()
. I took that from this answer.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With