public class Dates {
private static List< CustomDate > list = new ArrayList<>( );
public static void main( String[] args ) {
Map< LocalDate, List< LocalTime > > result2 = list.stream()
.map( CustomDate::getDatetime )
.collect(
HashMap::new,
Dates::accept,
Dates::merge
);
}
private static void accept( HashMap< LocalDate, List< LocalTime > > map, LocalDateTime date ) {
map.putIfAbsent( date.toLocalDate( ), new ArrayList<>())
.add( date.toLocalTime( ) );
}
private static void merge( Map< LocalDate, List< LocalTime > > map1, Map< LocalDate, List< LocalTime > > map2 ) {
map2.forEach( (date,timeList) -> {
map1.merge( date, timeList, ( l1, l2 ) -> {
l1.addAll( l2 );
return l1;
} );
});
}
}
或者简单地说:
Map< LocalDate, List< LocalTime > > result2 = list.stream()
.map( CustomDate::getDatetime )
.collect(
HashMap::new,
(map,date) -> {
map.putIfAbsent( date.toLocalDate( ), new ArrayList<>( )).add( date.toLocalTime( ) );
},
(map1, map2) -> {
map2.forEach( (date,timeList) -> {
map1.merge( date, timeList, ( l1, l2 ) -> {
l1.addAll( l2 );
return l1;
} );
});
}
);
不像调用 3 次收集器静态方法那样以声明方式执行此操作,可以让您更多地控制来指示行为,例如为数组使用初始大小、使用特定列表等,而不会丢失来自声明性的糖风格。
Map< LocalDate, List< LocalTime > > result2 = list.stream()
.map( CustomDate::getDatetime )
.collect(
TreeMap::new,
(map,date) -> {
map.putIfAbsent( date.toLocalDate( ), new ArrayList<>( 800 )).add( date.toLocalTime( ) );
},
(map1, map2) -> {
map2.forEach( (date,timeList) -> {
map1.merge( date, timeList, ( l1, l2 ) -> {
l1.addAll( l2 );
return l1;
} );
});
}
);
当然,如果默认行为是可以接受的,那就更简单了:
Map<LocalDate, List<LocalTime>> result = list.stream()
.map( CustomDate::getDatetime )
.collect(groupingBy(LocalDateTime::toLocalDate,
mapping(LocalDateTime::toLocalTime,toList())));
另一种方法是定义自己的收集器,这是我通常喜欢的。通过这样做,您可以获得控制权和以声明方式对逻辑进行编码并在需要时重用的能力。
这会将您的代码减少到:
Map< LocalDate, List< LocalTime > > result2 = list.stream()
.collect(
customDateCollector()
);
如果你有类似的东西:
class CustomDateCollector implements Collector<CustomDate, Map< LocalDate, List< LocalTime > >, Map< LocalDate, List< LocalTime > >> {
@Override
public Supplier< Map< LocalDate, List< LocalTime > > > supplier( ) {
return HashMap::new;
}
@Override
public BiConsumer< Map< LocalDate, List< LocalTime > >, CustomDate > accumulator( ) {
int goodStartUsingProblemKnowledge = 30;
return (map, customDate) -> map.putIfAbsent( customDate.getDatetime().toLocalDate(), new ArrayList<>( goodStartUsingProblemKnowledge) )
.add( customDate.getDatetime().toLocalTime() );
}
@Override
public BinaryOperator< Map< LocalDate, List< LocalTime > > > combiner( ) {
return (map1, map2) -> {
map2.forEach( (date,timeList) -> {
map1.merge( date, timeList, ( list1, list2 ) -> {
list1.addAll( list2 );
return list1;
} );
});
return map1;
};
}
@Override
public Function< Map< LocalDate, List< LocalTime > >, Map< LocalDate, List< LocalTime > > > finisher( ) {
return Function.identity();
}
@Override
public Set< Characteristics > characteristics( ) {
return Collections.unmodifiableSet( EnumSet.of( IDENTITY_FINISH, UNORDERED,CONCURRENT) );
}
public static CustomDateCollector customDateCollector(){
return new CustomDateCollector();
}
}