如何在不使用 forEach 的情况下对 Collectors.groupingBy() 之后的映射值进行分组

How to group map values after Collectors.groupingBy() without using forEach

提问人:Chidambar Shiragalle 提问时间:9/3/2020 最后编辑:Chidambar Shiragalle 更新时间:9/3/2020 访问量:1206

问:

我有一个 ProductDto 对象列表,我想使用 java 8 streams Collectors.groupingBy() 将它们分组为相似的对象。对记录进行分组后,我想将相似的记录合并为单个产品Dto。为了实现这一点,我使用了map.forEach并得到了预期的结果,但我想避免forEach循环,并想知道java 8中任何更好的解决方案。

下面是我的主类代码片段。

public class GroupTest {

public static void main(String[] args) {
    GroupTest t=new GroupTest();
    List<ProductDto> inputDtos=t.createInputData();
    List<ProductDto> resultDtos=t.getGroupedResult(inputDtos);
    //writing to Excel
}

private List<ProductDto> getGroupedResult(List<ProductDto> inputDtos) {
    
    Map<Object, List<ProductDto>> groupedMap = inputDtos.stream()
            .collect(Collectors.groupingBy(ProductDto::groupSimilarProductIdentifier));
    
    List<ProductDto> resultProductDtos=new ArrayList<>();

    groupedMap.forEach((key, dtos) -> {
        if (dtos.size() > 1) {
            ProductDto productDto = dtos.get(0);
            dtos.forEach(dto -> {
                if (dto.getLap1() != null) {
                    productDto.setLap1(dto.getLap1());
                } else if (dto.getLap2() != null) {
                    productDto.setLap2(dto.getLap2());
                } else if (dto.getLap3() != null) {
                    productDto.setLap3(dto.getLap3());
                }
            });
            resultProductDtos.add(productDto);
        } else {
            resultProductDtos.addAll(dtos);
        }
    });
    
    return resultProductDtos;
}

private List<ProductDto> createInputData(){
    List<ProductDto> dtos=new ArrayList<>();
    dtos.add(new ProductDto(1L,"DELL",8,"DELL_s001",null,null));
    dtos.add(new ProductDto(1L,"DELL",8,null,"DELL_s002",null));
    dtos.add(new ProductDto(1L,"DELL",8,null,null,"DELL_s003"));
    dtos.add(new ProductDto(1L,"HP",8,"HP_s001",null,null));
    dtos.add(new ProductDto(2L,"APPLE",16,"MAC_s001",null,null));
    return dtos;
}
}

这是 ProductDto 类代码

public class ProductDto {
private Long userId;
private String manufacter;
private int ram;
private String lap1;
private String lap2;
private String lap3;

public ProductDto(Long userId, String manufacter, int ram, String lap1, String lap2, String lap3) {
    super();
    this.userId = userId;
    this.manufacter = manufacter;
    this.ram = ram;
    this.lap1 = lap1;
    this.lap2 = lap2;
    this.lap3 = lap3;
}
//getters and Setters

public List<Object> groupSimilarProductIdentifier() {
    return Arrays.asList(userId, manufacter, ram);
}
}

下图是显示输入和输出记录的屏幕截图。输出记录正是我想要的结果。Java 8 中任何有效的替代或更好的解决方案都是非常受欢迎的。enter image description here

java-8 java-stream 收集器 lightweight-stream-api

评论

1赞 Eklavya 9/3/2020
你可以做,但你不能跳过forEach里面的操作,这是一个合并两个产品并返回合并产品的功能return new ArrayList<>(inputDtos.stream() .collect(Collectors.toMap(ProductDto::groupSimilarProductIdentifier, e -> e, (a,b) -> mergetwoProduct(a,b))).values());mergetwoProduct
0赞 Eklavya 9/3/2020
或。。。您也可以只使用地图功能而没有喜欢mergegroupingByMap<Object, ProductDto> groupedMap = new HashMap<>(); inputDtos.forEach(e -> groupedMap.merge(e.groupSimilarProductIdentifier(), e, (a,b) -> mergetwoProduct(a,b)));
0赞 Chidambar Shiragalle 9/3/2020
@Rono这是真的,我已经取出了 forEach 中的逻辑来合并两个Product 函数。以上解决方案有效。多谢。

答:

1赞 Chidambar Shiragalle 9/3/2020 #1

Rono 评论之后,我找到了答案,因此发布了我在 getGroupedResult 方法中所做的答案,并添加了一个新函数 mergetwoProduct。所以这可能会对某人有所帮助。 下面是更改后的 getGroupedResult 和 mergetwoProduct 方法的代码。

private List<ProductDto> getGroupedResult(List<ProductDto> inputDtos) {
    List<ProductDto> productdtos= new ArrayList<>(inputDtos.stream().collect(
            Collectors.toMap(ProductDto::groupSimilarProductIdentifier, e -> e, (a, b) -> mergetwoProduct(a, b)))
            .values());
    return productdtos;
}

private ProductDto mergetwoProduct(ProductDto p1,ProductDto p2) {
    if (p2.getLap1() != null) {
        p1.setLap1(p2.getLap1());
    } else if (p2.getLap2() != null) {
        p1.setLap2(p2.getLap2());
    } else if (p2.getLap3() != null) {
        p1.setLap3(p2.getLap3());
    }
    return p1;
}

评论

1赞 Eklavya 9/3/2020
另外,请记住您正在更改内部的数据。您可以通过创建一个新对象并将这两个对象合并到该对象中并返回来避免这种情况inputDtosmergetwoProduct