lambda表达式 —— 过滤再排序未生效问题排查

背景

项目中有个场景,需要将一个列表先按要求过滤,再根据某字段排序。图方便使用lambda表达式修改原列表将过滤和排序逻辑写到一行,打印输出列表发现过滤未生效。

代码示例:

java 复制代码
        List<Long> productIdList = Arrays.asList(1L);
        List<ProductDTO> productDTOList = new ArrayList<>();
        ProductDTO r1 = new ProductDTO();
        r1.setProductId(4L);
        r1.setPrice(100);
        productDTOList.add(r1);

        ProductDTO r2 = new ProductDTO();
        r2.setProductId(1L);
        r2.setPrice(20);
        productDTOList.add(r2);

        ProductDTO r3 = new ProductDTO();
        r3.setProductId(1L);
        r3.setPrice(100);
        productDTOList.add(r3);

        ProductDTO r4 = new ProductDTO();
        r4.setProductId(2L);
        r4.setPrice(150);
        productDTOList.add(r4);


        Map<Long, List<ProductDTO>> map = new HashMap<>();
        map.put(111L, productDTOList);

第一种 过滤+排序

java 复制代码
      map.forEach((infoId, productList) -> {
            productList.stream().filter(e -> productIdList.contains(e.getProductId())).collect(Collectors.toList()).sort(Comparator.comparing(ProductDTO::getPrice).reversed());

            System.out.println("=================" + productList);
        });

输出结果:

bash 复制代码
=================[ProductDTO(productId=4, price=100, name=null), ProductDTO(productId=1, price=20, name=null), ProductDTO(productId=1, price=100, name=null), ProductDTO(productId=2, price=150, name=null)]

第二种 过滤排序拆两行

java 复制代码
      map.forEach((infoId, productList) -> {
            productList = productList.stream().filter(e -> productIdList.contains(e.getProductId())).collect(Collectors.toList());
            productList.sort(Comparator.comparing(ProductDTO::getPrice).reversed());

            System.out.println("=================" + productList);
        });

输出结果:

bash 复制代码
=================[ProductDTO(productId=1, price=100, name=null), ProductDTO(productId=1, price=20, name=null)]

从上面示例可以看出第二种是我们需要的,第一种过滤没生效。

分析

针对下面这行代码,分析其运行过程。

java 复制代码
productList.stream().filter(e -> productIdList.contains(e.getProductId())).collect(Collectors.toList()).sort(Comparator.comparing(ProductDTO::getPrice).reversed());
  • stream()
    productList被转换成一个Stream,以声明性方式处理数据集合;
  • filter(e -> productIdList.contains(e.getProductId()))
    Stream上的filter方法被调用,该方法接受一个返回boolean值的函数作为参数。它检查productList中的每一个元素e的productId是否包含在productIdList中,只有满足条件的元素会被保留在Stream中。
  • collect(Collectors.toList())
    经过filter操作后,Stream中的元素被收集到一个新的List中。
  • sort(Comparator.comparing(ProductDTO::getPrice).reversed()):
    新生成的List被排序 。sort方法接受一个Comparator作为参数,用于定义排序规则。在这里,Comparator.comparing(ProductDTO::getPrice)创建了一个Comparator,它根据ProductDTO对象的getPrice方法返回的价格值进行排序。.reversed()表示降序排列。

综上,List 接口的 sort 方法使用了自然排序或者通过提供的 Comparator 进行排序,是对原始表进行排序,不会创建一个新的已排序的列表。在上述示例中,对collect(Collectors.toList())后生成的新列表进行排序,但新列表并未输出,对原列表productList无影响,既未过滤也未排序。

所以,上面第二种方案是先过滤再排序最后生成新列表返回,只不过原列表productList变量不再使用,指向新列表。也可以使用下面的写法。

java 复制代码
List<ProductDTO> list = productList.stream().filter(e -> productIdList.contains(e.getProductId())).sorted((Comparator.comparing(ProductDTO::getPrice).reversed())).collect(Collectors.toList());

此时list和原有productList为两个不同的列表,输出结果也是我们需要的。

另外productList = productList.stream()...collect(Collectors.toList()); 这种写法将新生成的列表赋值给了productList变量,productList指向了一个新的列表,但不会改变productList内容。如果原始的productList列表没有其他引用指向它,被其他使用,之后会被垃圾回收器回收。

原变量不再使用可以这么写,但会使代码不够清晰甚至引用出现失误,可以创建一个新的变量来存储新列表。

相关推荐
卓码软件测评1 小时前
第三方软件测试机构【性能测试工具用LoadRunner还是JMeter?】
java·功能测试·测试工具·jmeter·性能优化
钢门狂鸭4 小时前
关于rust的crates.io
开发语言·后端·rust
Lionel_SSL5 小时前
《深入理解Java虚拟机》第三章读书笔记:垃圾回收机制与内存管理
java·开发语言·jvm
记得开心一点嘛5 小时前
手搓Springboot
java·spring boot·spring
老华带你飞5 小时前
租房平台|租房管理平台小程序系统|基于java的租房系统 设计与实现(源码+数据库+文档)
java·数据库·小程序·vue·论文·毕设·租房系统管理平台
独行soc5 小时前
2025年渗透测试面试题总结-66(题目+回答)
java·网络·python·安全·web安全·adb·渗透测试
脑子慢且灵6 小时前
[JavaWeb]模拟一个简易的Tomcat服务(Servlet注解)
java·后端·servlet·tomcat·intellij-idea·web
华仔啊7 小时前
SpringBoot 中 6 种数据脱敏方案,第 5 种太强了,支持深度递归!
java·后端
异常驯兽师7 小时前
Spring 中处理 HTTP 请求参数注解全解析
java·spring·http
连合机器人8 小时前
晨曦中的守望者:当科技为景区赋予温度
java·前端·科技