Java中用Stream流取出分组后每组最大值对象的ID

取出分组后每组最大值对象的ID

如果只需要获取这些对象的ID(或其他特定字段),而不是整个对象,可以采用以下方法:

方法1:先获取对象再提取ID

java 复制代码
List<String> customerIds = orders.stream()
    .collect(Collectors.groupingBy(
        Order::getCustomerId,
        Collectors.maxBy(Comparator.comparing(Order::getOrderDate))
    ))
    .values().stream()
    .filter(Optional::isPresent)
    .map(Optional::get)
    .map(Order::getCustomerId) // 提取ID字段
    .collect(Collectors.toList());

方法2:直接在收集时提取ID(更高效)

java 复制代码
List<String> customerIds = orders.stream()
    .collect(Collectors.groupingBy(
        Order::getCustomerId,
        Collectors.collectingAndThen(
            Collectors.maxBy(Comparator.comparing(Order::getOrderDate)),
            opt -> opt.map(Order::getCustomerId).orElse(null)
        )
    ))
    .values().stream()
    .filter(Objects::nonNull)
    .collect(Collectors.toList());

方法3:使用toMap收集器(简洁版)

java 复制代码
List<String> customerIds = new ArrayList<>(
    orders.stream()
        .collect(Collectors.toMap(
            Order::getCustomerId,
            Function.identity(),
            (o1, o2) -> o1.getOrderDate().isAfter(o2.getOrderDate()) ? o1 : o2
        ))
        .values()
).stream()
.map(Order::getCustomerId)
.collect(Collectors.toList());

完整示例

java 复制代码
List<Order> orders = Arrays.asList(
    new Order("C1", LocalDate.of(2023, 1, 10), 100.0),
    new Order("C1", LocalDate.of(2023, 2, 15), 150.0),
    new Order("C2", LocalDate.of(2023, 1, 5), 200.0),
    new Order("C2", LocalDate.of(2023, 3, 20), 250.0),
    new Order("C3", LocalDate.of(2023, 2, 1), 300.0)
);

// 获取每个最新订单对应的客户ID
List<String> latestOrderCustomerIds = orders.stream()
    .collect(Collectors.groupingBy(
        Order::getCustomerId,
        Collectors.maxBy(Comparator.comparing(Order::getOrderDate))
    ))
    .values().stream()
    .flatMap(opt -> opt.map(Stream::of).orElseGet(Stream::empty))
    .map(Order::getCustomerId)
    .collect(Collectors.toList());

System.out.println("拥有最新订单的客户ID: " + latestOrderCustomerIds);

输出结果

复制代码
   拥有最新订单的客户ID: [C1, C2, C3]

关键点说明

  1. 如果只需要ID而不需要整个对象,建议使用方法2,它更高效
  2. map(Order::getCustomerId) 是提取ID的关键操作
  3. 方法3使用toMap合并函数,适合简单的最大/最小值场景
  4. 根据需求可以替换getCustomerId()为任何其他字段的getter方法
相关推荐
我学上瘾了13 分钟前
Spring Cloud的前世今生
后端·spring·spring cloud
波波0071 小时前
ASP.NET Core 健康检查实战:不只是一个 /health 接口
后端·asp.net
小码哥_常1 小时前
Spring Boot 搭建邮件发送系统:开启你的邮件自动化之旅
后端
石榴树下的七彩鱼2 小时前
图片修复 API 接入实战:网站如何自动去除图片水印(Python / PHP / C# 示例)
图像处理·后端·python·c#·php·api·图片去水印
我叫黑大帅2 小时前
为什么TCP是三次握手?
后端·网络协议·面试
我叫黑大帅3 小时前
如何排查 MySQL 慢查询
后端·sql·面试
techdashen3 小时前
Rust项目公开征测:Cargo 构建目录新布局方案
开发语言·后端·rust
一 乐3 小时前
电影院|基于springboot + vue电影院购票管理系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·电影院购票管理管理系统
恼书:-(空寄3 小时前
JVM GC 日志分析 + 常见 GC 场景 + 实战参数调优
java·jvm
消失的旧时光-19433 小时前
Spring Boot 实战(五):接口工程化升级(统一返回 + 异常处理 + 错误码体系 + 异常流转机制)
java·spring boot·后端·解耦