利用JDK8的Stream进行不同多层嵌套GroupBy及其性能对比

本文介绍了如何利用JDK8的Stream处理多层嵌套GroupBy的需求,通过背景介绍和测试类展示了不同分组实现方式,包括多字段Group By、多层嵌套Group By和遍历后Group By。通过对测试结果的分析,指出在数据量增大时,特定的分组方式能显著提高效率,并提出后续对源码分析的思考。
摘要由CSDN通过智能技术生成

背景

前段时间,接到一个sku摆放的需求,要求sku划分到指定的位置。因此,需要我们对每一条原始的sku数据按位置进行分组展示(见originSku)。其中位置是多层次划分,大致划分为:区域 -> 货架 -> 分区 -> 具体位置 -> 商品明细(如下图所示)。

在这里插入图片描述

本次需求涉及多层嵌套分组,为了保证执行效率,本次采用方案是按需查询所有的sku,在内存中进行分组处理。对于内存处理分组,我们很自然想到了JDK 8的新特性Stream,利用Stream下的Group By可以很好完成本次功能。

因为涉及的嵌套分组过多,达到4次,当sku数量增多时,这里很快就会出行性能问题,因此不得不考虑分组效率的问题。本次列举了3种分组的实现方式,并通过JMH工具进行性能测试,得到多层嵌套分组的最佳实践。

  • Stream 多字段 group by

  • 多层嵌套 group by

  • group by 后遍历再 group by

测试类

@BenchmarkMode(Mode.SingleShotTime)  // 测量调用1次耗时
@Warmup(iterations = 2)         		// 预热2次,避免JIT机制对结果进行干扰
@Threads(2)     // 2个线程,由执行环境cpu数量而定,本次电脑cpu核心数为8
@Fork(2)        //  fork 出两个进程
@OutputTimeUnit(TimeUnit.MILLISECONDS) // 单位 ms
public class MultiGroupByTest {
   

    private static List<OriginSku> originSkuList = getOriginSku();

    public static List<OriginSku> getOriginSku() {
   
        // 数据准备: 10 个区域, 区域下有 20 个货架, 货架下有 30 个分区, 分区下有 30 个位置, 每个位置放 10 sku
        List<OriginSku> originSkuList = new ArrayList<>();
        for (int i = 0; i < 10; i++) {
   
            for (int j = 0; j < 20; j++) {
   
                for (int k = 0; k < 30; k++) {
   
                    for (int l = 0; l < 30; l++) {
   
                        for (int m = 0; m < 10; m++) {
   
                            OriginSku sku = new OriginSku();
                            sku.setAreaNo("AreaNo-" + i);
                            sku.setShelfNo("ShelfNo-" + j);
                            sku.setPartitionNo("Partition-" + k);
                            sku.setLocationNo("LocationNo-" + l);
                            sku.setSkuCode(i + "-SkuCode-" + m);
                            sku.setSkuName(i + "-SkuName-" + m);
                            sku.setSkuQty(new BigDecimal(m));
                            sku.setOrder(m);
                            originSkuList.add(sku);
                        }
                    }
                }
            }
        }
        return originSkuList;
    }

    public static void main(String[] args) throws RunnerException {
   
        Options options = new OptionsBuilder()
                .include(MultiGroupByTest.class.getSimpleName())
                .build();
        new Runner(options).run();
    }
}

Stream 多字段 group by

先多字段group by,之后遍历组装数据。

@Benchmark
public List<AreaVo> buildSkuLocation3() {
   
    Map<String, Map<String, Map<String, Map<String, List<OriginSku>>>>> map = getOriginSku()
            .stream()
            .
  • 4
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值