Java использует потоки для получения количества заказанных продуктов по месяцам - PullRequest
0 голосов
/ 14 октября 2018

Я хотел бы подготовить простой отчет, который показывает количество заказанных продуктов по месяцам, как это (в порядке убывания):

Month         Number of products
2018-10       8
2018-11       3 

Мои данные:

  class Orders {

        private List<Order> orders = new ArrayList<>();

        public void prepareData() {

            Product socks = new ProductBuilder()
                    .setPrice(new BigDecimal("23"))
                    .setCategory(Category.C)
                    .setName("SOCKS")
                    .build();

            Product jacket = new ProductBuilder()
                    .setPrice(new BigDecimal("199"))
                    .setCategory(Category.A)
                    .setName("JACKET")
                    .build();

            Product watch = new ProductBuilder()
                    .setPrice(new BigDecimal("100"))
                    .setCategory(Category.B)
                    .setName("WATCH CASIO")
                    .build();


            Customer john = new CustomerBuilder()
                    .setAge(18)
                    .setName("JOHN")
                    .setSurname("JOHNSON")
                    .setEmail("john@johnson.com")
                    .build();

            Customer mike = new CustomerBuilder()
                    .setAge(20)
                    .setName("MIKE")
                    .setSurname("MAX")
                    .setEmail("mike@max.com")
                    .build();


            Order orderJohn = new OrderBuilder()
                    .setQuantity(2)
                    .setCustomer(john)
                    .setProduct(watch)
                    .setOrderDate(LocalDate.now())
                    .build();

            Order orderJohn2 = new OrderBuilder()
                    .setQuantity(4)
                    .setCustomer(john)
                    .setProduct(socks)
                    .setOrderDate(LocalDate.now())
                    .build();


             Order orderMike = new OrderBuilder()
                    .setQuantity(2)
                    .setCustomer(mike)
                    .setProduct(jacket)
                    .setOrderDate(LocalDate.now())
                    .build();

            orders.add(orderJohn);
            orders.add(orderJohn2);


            orders.add(orderMike);
    }
}

Теперь я используюgroupingBy чтобы получить номер заказа по месяцам

System.out.println(orders.stream()
       .collect(Collectors
               .groupingBy(e -> e.getOrderDate().withDayOfMonth(1), Collectors.counting())));

Но как добавить количество товаров по месяцам?Quantity поле в Order классе.

public class Order {

    private Customer customer;
    private Product product;
    private int quantity;
    private LocalDate orderDate;
   //get/set
}

public class Customer {

    private String name;
    private String surname;
    private int age;
    private String email;
    //get/set
}

public class Product {
    private String name;
    private BigDecimal price;
    private Category category;
    //get/set
}

И строители

public class CustomerBuilder {

    private Customer customer = new Customer();

    public CustomerBuilder setName(String name){
        customer.setName(name);
        return this;
    }

    public CustomerBuilder setSurname(String surname){
        customer.setSurname(surname);
        return this;
    }

    public CustomerBuilder setAge(int age){
        customer.setAge(age);
        return this;
    }

    public CustomerBuilder setEmail(String email){
        customer.setEmail(email);
        return this;
    }

    public Customer build()  {

        return customer;
    }

}
public class OrderBuilder {
   private Order order = new Order();

   public OrderBuilder setCustomer(Customer customer){
       order.setCustomer(customer);
       return this;
   }

   public OrderBuilder setProduct(Product product){
       order.setProduct(product);
       return this;
   }

   public OrderBuilder setQuantity(int quantity){
       order.setQuantity(quantity);
       return this;
   }

   public OrderBuilder setOrderDate(LocalDate orderDate){
       order.setOrderDate(orderDate);
       return this;
   }

   public Order build(){
       return order;
   }

}

public class ProductBuilder {

    private Product product = new Product();

    public ProductBuilder setCategory(Category category){
        product.setCategory(category);
        return this;
    }

    public ProductBuilder setName(String name){
        product.setName(name);
        return this;
    }

    public ProductBuilder setPrice(BigDecimal bigDecimal){
        product.setPrice(bigDecimal);
        return this;
    }

    public Product build() {
        return product;
    }
}

Ответы [ 2 ]

0 голосов
/ 14 октября 2018

Вы можете заменить Collectors.counting() на:

Collectors.summingInt(Order::getQuantity)

(работает так: Collectors.reducing(0, Order::getQuantity, Integer::sum))

РЕДАКТИРОВАТЬ:

Глядя на ваш комментарий к другому ответу, вот что вам нужно:

Map<LocalDate, IntSummaryStatistics> ord = orders.stream().collect(Collectors.groupingBy(e -> e.getOrderDate().withDayOfMonth(1), Collectors.summarizingInt(Order::getQuantity)));
ord.entrySet().stream().sorted(Map.Entry.comparingByValue((p1,p2) -> Long.compare(p2.getSum(), p1.getSum()))).forEach(System.out::println);
0 голосов
/ 14 октября 2018

Используйте Collectors.summingInt(Order::getQuantity) вместо Collectors.counting().

...