我们正在使用Java8流中的操作链处理事件流。作为处理的一部分,我们希望跟踪事件的数量及其状态,以便进行测试和监控。以下是我们的用例的简化示例,该用例打印给定日期流的星期几。
public class StreamStateHandling {
private static enum Status {RECEIVED, SUCCESS, ERROR};
private Map<Status,Integer> results = new EnumMap<>(Status.class);
private static final DateTimeFormatter formatter = DateTimeFormatter.ofPattern("MM/dd/yyyy");
private static Optional<LocalDate> parseDate(String dateString){
LocalDate localDate = null;
try {
localDate = LocalDate.from(formatter.parse(dateString));
}catch (DateTimeParseException e){
return Optional.empty();
}
return Optional.of(localDate);
}
private void doWork(){
Stream.of("12/31/2014",
"01-01-2015",
"12/31/2015",
"not a date",
"01/01/2016")
//.parallel()
.peek(v -> addResult(Status.RECEIVED))
.map(StreamStateHandling::parseDate)
.peek(v -> {if (!v.isPresent()) addResult(Status.ERROR);})
.filter(Optional::isPresent)
.map(Optional::get)
.map(DayOfWeek::from)
.peek(v -> addResult(Status.SUCCESS))
.forEach(System.out::println);
System.out.println(results);
}
public static void main(String args[]) {
new StreamStateHandling().doWork();
}
private void addResult(Status status){
int current = results.getOrDefault(status, 0);
results.put(status, current + 1);
}
}
基本上我们正在跟踪地图中的状态计数。这在单线程处理中工作正常,但在并行流中产生非确定性输出。
在现实世界中,我们有几种状态和操作链。总的来说,检测流程和跟踪进度的最佳方法是什么?更喜欢vanilla Java8实现,但如果使用开源库更容易,那就没问题。
非常感谢您的帮助。
答案 0 :(得分:3)
EnumMap
不是线程安全的,addResult()
中的读 - 修改 - 写逻辑也不是。尝试使用原子ConcurrentHashMap.merge()
来增加计数:
private Map<Status, Integer> results = new ConcurrentHashMap<>();
private void addResult(Status status) {
results.merge(status, 1, Integer::sum);
}