我有以下示例数据集,我想根据方向值使用Java流api进行变换/归约
Direction int[]
IN 1, 2
OUT 3, 4
OUT 5, 6, 7
IN 8
IN 9
IN 10, 11
OUT 12, 13
IN 14
到
Direction int[]
IN 1, 2,
OUT 3, 4, 5, 6, 7
IN 8, 9, 10, 11
OUT 12, 13
IN 14
到目前为止我写的代码
enum Direction { IN, OUT }
class Tuple {
Direction direction;
int[] data;
public Tuple merge(Tuple t) {
return new Tuple(direction, concat(getData(), t.getData()));
}
}
private static int[] concat(int[] first, int[] second) {
int[] result = Arrays.copyOf(first, first.length + second.length);
System.arraycopy(second, 0, result, first.length, second.length);
return result;
}
List<Tuple> reduce = tupleStream.reduce(new ArrayList<>(), WDParser::add, WDParser::combine);
private static List<Tuple> combine(List<Tuple> list1, List<Tuple> list2) {
System.out.println("combine");
list1.addAll(list2);
return list1;
}
private static List<Tuple> add(List<Tuple> list, Tuple t) {
System.out.println("add");
if (list.size() == 0) {
list.add(t);
} else if (list.size() > 0) {
int lastIndex = list.size() - 1;
Tuple last = list.get(lastIndex);
if (last.getDirection() == t.getDirection())
list.set(lastIndex, last.merge(t));
else
list.add(t);
}
return list;
}
我相信有比这更好和更简单的选择。
我发现的Java流api减少/组合在线示例和博客仅使用Integer :: sum函数。希望可以针对更复杂的案例进行构建。
答案 0 :(得分:4)
我认为您的解决方案已经非常不错了,尤其是与使用共享容器进行共享相比,使用归约简化了并行性。但是使用collect
比reduce
更容易如Holger所指出的那样。此外,累加器中的条件可以简化一些,而您忘记了合并合并器中的最后一个元素和第一个元素:
List<Tuple> reduce = tupleStream.collect(ArrayList::new, WDParser::add, WDParser::combine);
private static List<Tuple> combine(List<Tuple> list1, List<Tuple> list2)
{
if (!list2.isEmpty())
{
add(list1, list2.remove(0)); // merge lists in the middle if necessary
list1.addAll(list2); // add all the rest
}
return list1;
}
private static List<Tuple> add(List<Tuple> list, Tuple t)
{
int lastIndex = list.size() - 1;
if (list.isEmpty() || list.get(lastIndex).getDirection() != t.getDirection())
{
list.add(t);
}
else
{
list.set(lastIndex, list.get(lastIndex).merge(t));
}
return list;
}
您甚至可以使用LinkedList
和方法add/removeFirst/Last()
来代替使用索引来访问第一个/最后一个元素。
答案 1 :(得分:3)
如何?首先定义一个小的辅助方法:
private static Tuple mergeTwo(Tuple left, Tuple right) {
int[] leftArray = left.getData();
int[] rightArray = right.getData();
int[] result = new int[leftArray.length + rightArray.length];
System.arraycopy(leftArray, 0, result, 0, leftArray.length);
System.arraycopy(rightArray, 0, result, leftArray.length, rightArray.length);
return new Tuple(left.getDirection(), result);
}
我猜这接近您的concat/merge
,但只有一个。基本上是将两个Tuple
合并在一起的一种方式。
还有一种用于生成所需Collector
的辅助方法,您可以将其放入实用程序中以便重新使用:
private static Collector<Tuple, ?, List<Tuple>> mergedTuplesCollector() {
class Acc {
ArrayDeque<Tuple> deque = new ArrayDeque<>();
void add(Tuple elem) {
Tuple head = deque.peek();
if (head == null || head.getDirection() != elem.getDirection()) {
deque.offerFirst(elem);
} else {
deque.offerFirst(mergeTwo(deque.poll(), elem));
}
}
Acc merge(Acc right) {
Tuple lastLeft = deque.peekLast();
Tuple firstRight = right.deque.peekFirst();
if (lastLeft.getDirection() == firstRight.getDirection()) {
deque.offerLast(mergeTwo(deque.pollLast(), right.deque.pollFirst()));
} else {
deque.addAll(right.deque);
}
return this;
}
public List<Tuple> finisher() {
return new ArrayList<>(deque);
}
}
return Collector.of(Acc::new, Acc::add, Acc::merge, Acc::finisher);
}
用法例如:
List<Tuple> merged = tuples.stream()
.parallel()
.collect(mergedTuplesCollector());
答案 2 :(得分:1)
这是一种使用略有不同的数据结构的替代方法。
如果这是一个选项,则从int[]
更改为List<Integer>
可提供更大的灵活性(更不用说避免多次创建/复制数组了):
class Tuple {
Direction direction;
List<Integer> data;
}
以下功能对Deque
集合进行合并:
private static List<Integer> next(Deque<Tuple> t, Direction d) {
if (!t.isEmpty() && t.peekLast().getDirection() == d) {
return t.peekLast().getData();
} else {
Tuple next = new Tuple();
next.direction = d;
next.data = new ArrayList<>();
t.addLast(next);
return next.data;
}
}
有了它,流看起来就像:
Deque<Tuple> deq = new LinkedList<>(); //the final collection of tuples
tuples.stream()
.flatMap(tp -> tp.getData().stream()
.map(d -> Pair.of(tp.getDirection(), Integer.valueOf(d))))
.forEach(el -> next(deq, el.getLeft()).add(el.getRight()));
答案 3 :(得分:0)
关于这个话题,我有两个想法。第一个是获取this answer中的索引并将其相应地分组。
第二个想法-如果您已经拥有Stream
,则应使用自定义Collector
(类似于其他解决方案,尽管使用的是Deque
):
private Collector<Tuple, ?, List<Tuple>> squashTuples() {
return new Collector<Tuple, Deque<Tuple>, List<Tuple>>() {
@Override
public Supplier<Deque<Tuple>> supplier() {
return ArrayDeque::new;
}
@Override
public BiConsumer<Deque<Tuple>, Tuple> accumulator() {
return (acc, e) -> {
Objects.requireNonNull(e);
if (!acc.isEmpty() && acc.peekLast().getDirection() == e.getDirection()) {
acc.offerLast(acc.pollLast().merge(e));
} else {
acc.offerLast(e);
}
};
}
@Override
public BinaryOperator<Deque<Tuple>> combiner() {
return (left, right) -> {
if (!left.isEmpty() && !right.isEmpty() && left.peekLast().getDirection() == right.peekFirst().getDirection()) {
left.offerLast(left.pollLast().merge(right.pollFirst()));
}
left.addAll(right);
return left;
};
}
@Override
public Function<Deque<Tuple>, List<Tuple>> finisher() {
return ArrayList::new;
}
@Override
public Set<Characteristics> characteristics() {
return EnumSet.noneOf(Characteristics.class);
}
};
}