java如何利用流式计算筛选出同一天时间最新的一条数据

有时候用sql做自联接或者groupBy不方便,可以选择java的流式计算

两种方案

1:

// 身体肌肉率 MuscleList<KpiVO> lastMuscle = allDevicesVOS.stream().filter(device -> StringUtil.isNotBlank(device.getMuscle())) // 过滤出身体肌肉率都不为 null 的记录.collect(Collectors.groupingBy(device -> device.getCheckTime().toLocalDate())) // 按 LocalDate 分组.values().stream().map(devices -> devices.stream().max((d1, d2) -> d1.getCheckTime().compareTo(d2.getCheckTime())) // 找到每组最新的记录.orElse(null)) // 如果某组为空,返回 null.filter(device -> device != null) // 过滤掉 null 值.map(device -> {KpiVO kpiVO = new KpiVO();kpiVO.setCheckTimes(device.getCheckTime().toLocalDate().toString());kpiVO.setLast(device.getMuscle());return kpiVO;}).sorted((k1, k2) -> k1.getCheckTimes().compareTo(k2.getCheckTimes())) // 按 checkTimes 正序排列.collect(Collectors.toList());// 转换回 List<KpiVO>

2:

// 使用 Java 8 Stream API 处理结果List<LocalDate> result = checkTimes.stream().collect(Collectors.groupingBy(e -> e.getCheckTime().toLocalDate(),  // 按日期分组Collectors.maxBy((e1, e2) -> e1.getCheckTime().compareTo(e2.getCheckTime()))  // 选择每个日期中时间最晚的记录)).values().stream().map(Optional::get).sorted(Comparator.comparing(NormBasicDeviceDataEntity::getCheckTime).reversed())  // 对日期进行降序排序.map( entity -> {return entity.getCheckTime().toLocalDate();}).collect(Collectors.toList());

两种方案的区别在于

两段代码中都有groupingBy操作,特别是第二段如何使用maxBy来选择每个日期中的“最晚记录”。

第二段代码解析

 

List<LocalDate> result = checkTimes.stream() .collect(Collectors.groupingBy(e -> e.getCheckTime().toLocalDate(), // 按日期分组 Collectors.maxBy((e1, e2) -> e1.getCheckTime().compareTo(e2.getCheckTime())) // 选择每个日期中时间最晚的记录 )) .values() .stream() .map(Optional::get) .sorted(Comparator.comparing(NormBasicDeviceDataEntity::getCheckTime).reversed()) // 对日期进行降序排序 .map(entity -> { return entity.getCheckTime().toLocalDate(); }) .collect(Collectors.toList());

这段代码的核心是在分组操作中使用了maxBy,让我们一步一步来分析。

1. groupingBy + maxBy

 

Collectors.groupingBy(e -> e.getCheckTime().toLocalDate(), Collectors.maxBy((e1, e2) -> e1.getCheckTime().compareTo(e2.getCheckTime())))

  • 分组依据: 这部分和前面相同,e -> e.getCheckTime().toLocalDate()是按每个元素(checkTime)的日期部分(LocalDate)进行分组。

  • maxBy: 关键的区别在于Collectors.maxBy,它并不是简单地将所有元素放入一个集合中,而是用于选择每个分组中“最大”的元素,这个“最大”是根据你给定的比较器来确定的。

    • maxBy((e1, e2) -> e1.getCheckTime().compareTo(e2.getCheckTime()))表示按时间(checkTime)比较,并选择每个分组中时间最晚的那个元素。
    • 由于是按时间升序排序(compareTo方法的返回值),因此maxBy返回的将是每个日期中最新的记录

    结果会是一个Map<LocalDate, Optional<NormBasicDeviceDataEntity>>,其中每个键(日期)对应一个Optional<NormBasicDeviceDataEntity>,表示每个日期的最新设备数据记录。

2. .values().stream()

  • values()会返回一个Collection<Optional<NormBasicDeviceDataEntity>>,这个集合是按日期分组后,每个日期组内最新的NormBasicDeviceDataEntity对象。
  • 通过.stream()转化为流,可以继续后续操作。

3. .map(Optional::get)

  • Optional::get会从每个Optional<NormBasicDeviceDataEntity>中提取出实际的NormBasicDeviceDataEntity对象(假设每个Optional不为空)。

4. .sorted(Comparator.comparing(NormBasicDeviceDataEntity::getCheckTime).reversed())

  • 按照每个NormBasicDeviceDataEntitycheckTime降序排序(reversed()),确保结果是按照时间从最新到最旧排列。

5. .map(entity -> entity.getCheckTime().toLocalDate())

  • 最后将每个NormBasicDeviceDataEntitycheckTime转换为LocalDate,即提取出日期部分。

6. collect(Collectors.toList())

  • 最终,结果被收集到一个List<LocalDate>中。

主要区别:groupingBymaxBy的结合 vs. 单纯的groupingBy

  • groupingBy + maxBy:
    • 在每个分组内使用maxBy来找到最大(最新)的记录。这意味着对于每个日期,我们只保留时间最晚的记录,而不是所有记录。
    • 结果是一个Map<LocalDate, Optional<NormBasicDeviceDataEntity>>,每个日期对应一个Optional,表示日期下的最新设备数据。
  • 单纯的groupingBy:
    • 单纯的groupingBy只会将数据根据某个条件分组,不会对每组数据做任何筛选或操作。因此,它返回的是Map<LocalDate, List<Device>>,其中每个日期对应一个包含该日期所有设备的列表。
    • 没有maxBy的情况下,每个日期下的所有记录都会保留,并没有进行筛选。

总结对比

  • 第一种方法(使用groupingBy后再用max)对每个分组中的数据进行了筛选,选出每个日期下的最新记录。
  • 第二种方法(使用groupingBymaxBy)使用maxBy直接在分组时就筛选出每个日期组中的最新记录,避免了后续的手动筛选。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/16976.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

I/O文件:文件的关闭

int fclose(FILE *stream); 成功关闭返回1&#xff0c;关闭失败返回EOF即-1&#xff0c;并设置errno。 流关闭时自动刷新缓冲中的数据并释放缓冲区 当一个程序正常终止时&#xff0c;所有打开的流都会被关闭 流一旦关闭就不能执行任何操作。 运行结果&#xff1a; 若未成功打…

联邦学习的未来:深入剖析FedAvg算法与数据不均衡的解决之道

引言 随着数据隐私和数据安全法规的不断加强&#xff0c;传统的集中式机器学习方法受到越来越多的限制。为了在分布式数据场景中高效训练模型&#xff0c;同时保护用户数据隐私&#xff0c;联邦学习&#xff08;Federated Learning, FL&#xff09;应运而生。它允许多个参与方…

深入理解Flutter生命周期函数之StatefulWidget(一)

目录 前言 1.为什么需要生命周期函数 2.开发过程中常用的生命周期函数 1.initState() 2.didChangeDependencies() 3.build() 4.didUpdateWidget() 5.setState() 6.deactivate() 7.dispose() 3.Flutter生命周期总结 1.调用顺序 2.函数调用时机以及主要作用 4.生…

LoFTR: Detector-Free Local Feature Matching with Transformers

LoFTR: Detector-Free Local Feature Matching with Transformers 整体概括 Loftr特征点匹配算法与传统的特征点匹配算法的优势&#xff1a; 不需要先得到特征点&#xff0c;这也就解决了第一个问题End2End的方式&#xff0c;用起来比较方便&#xff0c;效果也更好 整体流程的…

免押租赁系统的优势与应用前景分析

内容概要 免押租赁系统是一种新兴的租赁形式&#xff0c;它利用了信用大数据与区块链技术的优势&#xff0c;帮助用户摆脱了传统租赁中常见的押金烦恼。通过这种方式&#xff0c;用户不仅可以体验到更低的租用门槛&#xff0c;还能享受到更顺畅的交易过程。用户只需提供基本的…

「Qt Widget中文示例指南」如何创建一个窗口标志?(二)

Qt 是目前最先进、最完整的跨平台C开发工具。它不仅完全实现了一次编写&#xff0c;所有平台无差别运行&#xff0c;更提供了几乎所有开发过程中需要用到的工具。如今&#xff0c;Qt已被运用于超过70个行业、数千家企业&#xff0c;支持数百万设备及应用。 窗口标志要么是类型…

(附项目源码)Java开发语言,211 springboot 在线问诊系统的设计与实现,计算机毕设程序开发+文案(LW+PPT)

摘 要 针对医院门诊等问题&#xff0c;对在线问诊进行研究分析&#xff0c;然后开发设计出在线问诊系统以解决问题。在线问诊系统主要功能模块包括首页、轮播图管理、公告信息管理、资源管理、系统用户管理&#xff08;管理员、患者用户、医生用户&#xff09;、模块管理&#…

Vulnhub靶场 Billu_b0x 练习

目录 0x00 准备0x01 主机信息收集0x02 站点信息收集0x03 漏洞查找与利用1. 文件包含2. SQL注入3. 文件上传4. 反弹shell5. 提权&#xff08;思路1&#xff1a;ssh&#xff09;6. 提权&#xff08;思路2&#xff1a;内核&#xff09;7. 补充 0x04 总结 0x00 准备 下载链接&#…

Linux应用层学习——Day2(文件IO)

fopen #include <stdio.h>int main(int argc,char const *argv[]) {// FILE *fopen (const char *__restrict __filename&#xff1a;字符串要打开的文件名称,// const char *__restrict __modes&#xff1a;打开的模式// 1) r&#xff1a;只读模式&#xff0c;如果…

Java并发篇--ReentrantLock原理

ReentrantLock原理 非公平锁实现原理 加锁解锁流程 先从构造器开始看,默认为非公平锁 public ReentrantLock() {sync new NonfairSync(); }NonfairSync 继承自 AQS 没有竞争时 第一个竞争出现时 Thread-1执行了 1.CAS尝试将state由0改为1,结果失败 2.进入tryAcquire逻辑…

分析http话术异常挂断原因

用户反馈在与机器人通话时&#xff0c;自己明明有说话&#xff0c;但是通话还是被挂断了&#xff0c;想知道原因。 分析日志 我们根据用户提供的freeswitch日志分析&#xff1a;发现是因为超时导致话术执行hangup动作&#xff0c;结束了通话。 从这一行向上分析日志&#xff…

【Pikachu】PHP反序列化RCE实战

痛是你活着的证明 1.PHP反序列化概述 在理解 PHP 中 serialize() 和 unserialize() 这两个函数的工作原理之前&#xff0c;我们需要先了解它们各自的功能及其潜在的安全隐患。接下来&#xff0c;我会对相关概念做更详细的扩展解释。 1. 序列化 serialize() 序列化&#xff…

零基础Java第十八期:图书管理系统

目录 一、package book 1.1. Book 1.2. BookList 二、package user 2.1. User 2.2. NormalUser与AdminiUser 三、Main 四、NormalUser与AdminiUser的菜单界面 五、package operation 5.1. 设计管理员菜单 六、业务逻辑 七、完整代码 今天博主来带大家实现一个…

【每日题解】3239. 最少翻转次数使二进制矩阵回文 I

给你一个 m x n 的二进制矩阵 grid 。 如果矩阵中一行或者一列从前往后与从后往前读是一样的&#xff0c;那么我们称这一行或者这一列是 回文 的。 你可以将 grid 中任意格子的值 翻转 &#xff0c;也就是将格子里的值从 0 变成 1 &#xff0c;或者从 1 变成 0 。 请你返回 …

vue面试题9|[2024-11-15]

问题1&#xff1a;scoped原理 1.作用&#xff1a;让样式在本组件中生效&#xff0c;不影响其他组件 2.原理&#xff1a;给节点新增自定义属性&#xff0c;然后css根据属性选择器添加样式。 问题2&#xff1a;让css只在当前组件生效 <style scoped> 问题3&#xff1a;scss…

2024新版pycharm如何切换anaconda虚拟环境

2024新版pycharm如何切换anaconda虚拟环境 不得不说这界面改的真不错&#xff0c;看着很舒服。 回归正题&#xff0c; 导入项目后点击文件>设置&#xff0c;找到解释器。 添加解释器>添加本地解释器 以前是选择conda环境&#xff0c;现在直接就是Virtualenv 环境 看…

Codeforces Round 987 (Div. 2)(前四道)

A. Penchick and Modern Monument 翻译&#xff1a; 在繁华大都市马尼拉的摩天大楼中&#xff0c;菲律宾最新的 Noiph 购物中心刚刚竣工&#xff01;建筑管理方 Penchick 订购了一座由 n 根支柱组成的先进纪念碑。 纪念碑支柱的高度可以用一个由 n 个正整数组成的数组 h 来表示…

探索AI驱动的企业知识库:提升管理效率的新利器

对于企业而言&#xff0c;如何高效管理知识、提升团队协作、加速决策过程&#xff0c;已成为生存与发展的关键。而人工智能(AI)的迅速发展为解决这些问题提供了新的思路和工具。越来越多的企业开始构建AI驱动的知识库&#xff0c;以实现信息的智能管理与利用。本文将深入探讨AI…

C语言项⽬实践-贪吃蛇

目录 1.项目要点 2.窗口设置 2.1mode命令 2.2title命令 2.3system函数 2.Win32 API 2.1 COORD 2.2 GetStdHandle 2.3 CONSOLE_CURSOR_INFO 2.4 GetConsoleCursorInfo 2.5 SetConsoleCursorInfo 2.5 SetConsoleCursorPosition 2.7 GetAsyncKeyState 3.贪吃蛇游戏设…

为什么 Vue3 封装 Table 组件丢失 expose 方法呢?

在实际开发中&#xff0c;我们通常会将某些常见组件进行二次封装&#xff0c;以便更好地实现特定的业务需求。然而&#xff0c;在封装 Table 组件时&#xff0c;遇到一个问题&#xff1a;Table 内部暴露的方法&#xff0c;在封装之后的组件获取不到。 代码展示为&#xff1a; …