结合两个流Java 8

Kev*_*vin 4 java java-8 java-stream

有没有办法结合以下两种语态?

    Map<Integer,Double> collX = listeAllerPunkte.stream().collect(groupingBy(DataPoint::getId,
                                averagingDouble(DataPoint::getX)));
    Map<Integer,Double> collY = listeAllerPunkte.stream().collect(groupingBy(DataPoint::getId,
                                averagingDouble(DataPoint::getY)));
Run Code Online (Sandbox Code Playgroud)

我有一个DataPoints这样的类:

public class DataPoint {

    public final double x;
    public final double y;
    private int Id;

    public DataPoint(double x, double y) {
        this.x = x;
        this.y = y;
    }

    public double getX() {
        return x;
    }

    public double getY() {
        return y;
    }

    public int getId() {
        return Id;
    }
}
Run Code Online (Sandbox Code Playgroud)

Id包含之间的随机值0-5.

listeAllerPunkte是一个List有很多DataPoints

现在我想为List中的DataPoint每个创建一个DataPoints相同的Id.本DataPoint应具有的平均x and y values数据点具有相同Id.

使用来自Beginning的两个Statemantes,我必须DataPoints手动创建两个Maps.有没有办法直接在流中创建它们?

Hol*_*ger 5

一般的解决方案是使用一个收集器,它可以将两个收集器组合在一起进行处理.不幸的是,标准API中不存在这样的收集器,但是这个答案提供了这种收集器的实现.

或者,您可以通过创建自己的类来保存点的摘要,为此特定情况创建解决方案,例如

static class DataPointSummary {
    long count;
    double sumX, sumY;

    public double getAverageX() {
        return count==0? 0: sumX/count;
    }
    public double getAverageY() {
        return count==0? 0: sumY/count;
    }
    public void add(DataPoint p) {
        count++;
        sumX+=p.getX();
        sumY+=p.getY();
    }
    public DataPointSummary merge(DataPointSummary s) {
        count+=s.count;
        sumX+=s.sumX;
        sumY+=s.sumY;
        return this;
    }
    @Override
    public String toString() {
        return "DataPointSummary["+count+" points"
            +", avg x="+getAverageX()+", avg y="+getAverageY()+']';
    }
}
Run Code Online (Sandbox Code Playgroud)

然后你可以收集你的积分

Map<Integer,DataPointSummary> coll = listeAllerPunkte.stream().collect(
    groupingBy(DataPoint::getId, Collector.of(
        DataPointSummary::new, DataPointSummary::add, DataPointSummary::merge)));
Run Code Online (Sandbox Code Playgroud)

请注意,我认为您的方法签名public double getId()是一个拼写错误,实际上public int getId(),您的问题中的示例将不起作用.

如果点的坐标具有相同的大小,则上面的摘要实现很有效.如果在同一组中遇到非常大的值和非常小的值,则可能需要使用误差补偿算法求和.我建议使用JRE的摘要实现,而不是自己实现它:

static class DataPointSummary {
    final DoubleSummaryStatistics x=new DoubleSummaryStatistics();
    final DoubleSummaryStatistics y=new DoubleSummaryStatistics();

    public double getAverageX() {
        return x.getAverage();
    }
    public double getAverageY() {
        return y.getAverage();
    }
    public void add(DataPoint p) {
        x.accept(p.getX());
        y.accept(p.getY());
    }
    public DataPointSummary merge(DataPointSummary s) {
        x.combine(s.x);
        y.combine(s.y);
        return this;
    }
    @Override
    public String toString() {
        return "DataPointSummary["+x.getCount()+" points"
            +", avg x="+getAverageX()+", avg y="+getAverageY()+']';
    }
}
Run Code Online (Sandbox Code Playgroud)

此变体的使用方式与第一个相同.