Я хотел бы сгруппировать коллекцию объектов Rental в установленный срок, но я хочу создать новый объект RentalReport для каждой группы с ключом в качестве предопределенного значения (enum) и группой, которая будет являться свойством этого объекта. Я достиг этого, подбирая коллекцию по каждому критерию и создавая объект RentalReport для каждого, но мне было интересно, можно ли это сделать с помощью метода groupingBy класса Collectors.
Можно ли сгруппировать по предопределенному набору фильтров в java 8, чтобы я мог создать карту, ключом которой является enum, а значением является коллекция объектов Rental. Затем я мог бы перебрать эту карту и сгенерировать объекты RentalReport.
Я создал эту демонстрацию, но реальная задача включает в себя несколько групп по предложениям, поэтому было бы здорово, если бы я мог добиться этого путем группировки.
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
import org.joda.time.DateTime;
public class TestGrouping {
enum RentalClassification {
UNRESTRICTED, //No restriction restricted=false
OVERDUE, //todays date after due date.
NEARLY_OVERDUE, // todays date after due date + 2 days
NOT_OVERDUE
}
public static void main(String[] args) {
class Rental {
Integer rentalId;
Integer productId;
boolean restricted;
DateTime dueDate;
public Rental(Integer rentalId, Integer productId, boolean restricted, DateTime dueDate){
this.rentalId = rentalId;
this.productId = productId;
this.restricted = restricted;
this.dueDate = dueDate;
}
public Integer getRentalId() {
return rentalId;
}
public void setRentalId(Integer rentalId) {
this.rentalId = rentalId;
}
public Integer getProductId() {
return productId;
}
public void setProductId(Integer productId) {
this.productId = productId;
}
public boolean isRestricted() {
return restricted;
}
public void setRestricted(boolean restricted) {
this.restricted = restricted;
}
public DateTime getDueDate() {
return dueDate;
}
public void setDueDate(DateTime dueDate) {
this.dueDate = dueDate;
}
public String toString(){
return "RentalId:"+this.rentalId+". ProductId:"+this.productId+". Due date:"+this.dueDate+". -";
}
}
class RentalReport {
RentalClassification classification;
List<Rental> rentals;
public RentalReport(RentalClassification classification, List<Rental> rentals) {
this.classification = classification;
this.rentals = rentals;
}
public RentalClassification getClassification() {
return classification;
}
public void setClassification(RentalClassification classification) {
this.classification = classification;
}
public List<Rental> getRentals() {
return rentals;
}
public void setRentals(List<Rental> rentals) {
this.rentals = rentals;
}
public String toString(){
StringBuilder sb = new StringBuilder("Classification:"+this.classification.name()+". Rental Ids:");
this.rentals.forEach(r -> sb.append(r.getRentalId()));
return sb.toString();
}
}
DateTime today = new DateTime();
List<Rental> rentals = Arrays.asList(
new Rental(1,100, true, today.plusDays(-10)),
new Rental(2,101, false, today.plusDays(-10)),
new Rental(3,102, true, today.plusDays(-4)),
new Rental(4,103, true, today.plusDays(-4)),
new Rental(5,104, true, today.plusDays(-4)),
new Rental(6,105, true, today.plusDays(2)),
new Rental(7,106, true, today.plusDays(2)),
new Rental(8,107, true, today.plusDays(2)),
new Rental(9,108, true, today.plusDays(4)),
new Rental(10,109, true, today.plusDays(5))
);
List<RentalReport> rentalReports = new ArrayList<RentalReport>();
List<Rental> unrestrictedRentals = rentals.stream()
.filter(r -> !r.isRestricted())
.collect(Collectors.toList());
rentalReports.add(new RentalReport(RentalClassification.UNRESTRICTED, unrestrictedRentals));
List<Rental> overdueRentals = rentals.stream()
.filter(r -> r.isRestricted() && r.getDueDate().isBefore(new DateTime()))
.collect(Collectors.toList());
rentalReports.add(new RentalReport(RentalClassification.OVERDUE, overdueRentals));
List<Rental> nearlyOverdueRentals = rentals.stream()
.filter(r -> r.isRestricted()
&& r.getDueDate().isAfter(new DateTime())
&& r.getDueDate().isBefore(new DateTime().plusDays(2)))
.collect(Collectors.toList());
rentalReports.add(new RentalReport(RentalClassification.NEARLY_OVERDUE, nearlyOverdueRentals));
List<Rental> notOverdueRentals = rentals.stream()
.filter(r -> r.isRestricted() && r.getDueDate().isAfter(new DateTime()))
.collect(Collectors.toList());
rentalReports.add(new RentalReport(RentalClassification.NOT_OVERDUE, notOverdueRentals));
System.out.println("Rental Reports: "+rentalReports.toString());
}
}