java 8 stream如何将2列映射到地图中

时间:2015-04-21 22:35:46

标签: java java-stream java-collections-api

我有一张桌子:

  

名称|数据| NUM
  item1 | 16 | 2
  item2 | 17 | 3
  item1 | 16 | 5

我想将其转换为:

  

{item1:{16 + 16,2 + 5},item2:{17,3}}

但是我只能成功地产生以下结果:

  

{item1:16 + 16,item2:17}

使用以下代码: Stats类存储两个字段,但我不知道如何将两个字段添加到数据

class Stats {
    public Integer data, num;
    public Stats(Integer td, Integer nf) {
        data = td;
        num = nf;
    }
    public Integer getdata(){
        return data;
    }
    public Integer getnum(){
        return num;
    }
}
Map<String, Stats> map = new HashMap<>();
Stream<String> lines = Files.lines(Paths.get(table));
map = lines.map(x -> 
             .collect(Collectors.toMap(
                z -> z[0],
                z -> Integer.parseInt(z[1]),
                (a, b) -> a+b));

上述代码仅适用于Map<String, Int>,因为我不确定如何使Map<String, Stats>有效。 知道如何将第二列上的项目放入地图,而只使用1个管道吗?

3 个答案:

答案 0 :(得分:1)

首先需要创建一个包含两个整数的类:

public final class TwoInts {
    private final int data;
    private final int num;

    public TwoInts(int data, int num) { 
        this.data = data;
        this.num = num;
    }

    // getters omitted for brevity
}

然后你可以重用已有的逻辑,除了使用整数之外,你将使用TwoInts实例。因此,您需要能够将两个TwoInts实例一起添加到另一个TwoInts实例中(就像您通过添加两个整数生成一个新整数一样):

public final class TwoInts {
    private final int data;
    private final int num;

    public TwoInts(int data, int num) { 
        this.data = data;
        this.num = num;
    }

    public static TwoInts sum(TwoInts first, TwoInts second) {
        return new TwoInts(first.data + second.data, 
                           first.num + second.num);
    }

    // getters omitted for brevity
}

然后你去了:

Stream<String> lines = Files.lines(Paths.get(table));
Map<String, TwoInts> map = 
    lines.map(line -> toArrayOfStrings(line))
         .collect(Collectors.toMap(
              row -> row[0],
              row -> new TwoInts(Integer.parseInt(row[1]), Integer.parseInt(row[2])),
              (a, b) -> TwoInts.sum(a, b));

答案 1 :(得分:0)

这是一个实现:

  • 不需要任何额外的课程
  • 适用于任何长度的数值(不仅仅是2)
  • (技术上)只是一行


Stream<String> lines = Stream.of("item1,16,2",
      "item2,17,3",
      "item1,16,5"); // sample input

Map<String, List<Integer>> map = lines.map(s -> s.split(","))
  .collect(Collectors.toMap(a -> a[0],
      a -> Arrays.asList(a).stream()
       .skip(1).map(Integer::parseInt)
        .collect(Collectors.toList()),
      (a, b) -> {for (int i = 0; i < a.size(); i++)
          a.set(i, a.get(i) + b.get(i)); return a;}));

运行此输出后打印地图:

{item2=[17, 3], item1=[32, 7]}

请参阅此代码的live demo

答案 2 :(得分:0)

首先,你可以让你的Stats类更加简化,添加合并方法(我也稍微美化了):

class Stats {
    final int data, num;

    public Stats() {
        data = num = 0;
    }

    public Stats(int data, int num) {
        this.data = data;
        this.num = num;
    }

    public Stats merge(Stats stats) {
        return new Stats(this.data+stats.data, this.num+stats.num);
    }

    public int getData(){
        return data;
    }
    public int getNum(){
        return num;
    }

    @Override
    public String toString() {
        return "{" + data + "," + num + "}";
    }
}

现在,您可以使用收集器链解决您的任务。假设你的输入来自字符串流,就像波希米亚的回答:

Stream<String> lines = Stream.of("item1,16,2",
          "item2,17,3",
          "item1,16,5"); // sample input

现在可以用这种方式构建生成的地图:

Map<String, Stats> map = lines.map(s -> s.split(",")).collect(
    Collectors.groupingBy(a -> a[0], 
    Collectors.mapping(a -> new Stats(Integer.parseInt(a[1]), Integer.parseInt(a[2])), 
    Collectors.reducing(new Stats(), Stats::merge))));
System.out.println(map);

现场演示是here