当前位置: 首页 > article >正文

Java8 Stream sorted排序实战:从Comparator基础到多级排序进阶

1. 从零开始理解Stream sorted排序第一次接触Java8的Stream sorted方法时我盯着那段链式调用的代码看了足足十分钟。就像刚拿到新手机的老人明明按键就在眼前却不知道从哪下手。后来在实际项目中踩过几次坑才明白sorted()本质上就是个智能排序器而Comparator就是告诉它排序规则的说明书。先看最简单的自然排序。假设我们有个字符串列表ListString names Arrays.asList(张三, 李四, 王五);用names.stream().sorted().forEach(System.out::println);就能按字典序输出。这就像把一堆杂乱的书本按书名首字母排列但现实中的需求往往更复杂。有次我处理用户数据时发现直接用sorted()抛出了异常——原来列表里的自定义对象没实现Comparable接口就像试图给一堆没有条形码的商品排序系统根本不知道谁该排在前面。这时候就需要祭出Comparator了。最基础的写法是用lambda表达式list.stream().sorted((o1, o2) - o1.getAge() - o2.getAge())。这个减号操作符就像天平当左边大于右边返回正数就是升序反过来就是降序。不过实际开发中我更喜欢用Comparator.comparing(User::getAge)这种写法代码更语义化就像直接告诉系统嘿按年龄排序。2. Comparator的七十二变Comparator的灵活之处在于它能像乐高积木一样组合。记得有次做电商项目商品要按销量降序排列销量相同的再按价格升序排。这用传统写法得嵌套多层if-else而用Stream sorted只需要products.stream() .sorted(Comparator.comparing(Product::getSales).reversed() .thenComparing(Product::getPrice)) .collect(Collectors.toList());这里的reversed()就像把排序规则倒过来看而thenComparing相当于说前面的规则分不出胜负时再用这个新规则。这种链式调用比俄罗斯套娃式的if-else清爽多了。更复杂的场景比如处理多语言排序时可以用Comparator.comparing(String::length, Comparator.reverseOrder())实现先按字符串长度降序再按字母顺序升序。这就像图书馆先按书厚度分大类再按书名细分。实测下来这种写法比传统Collections.sort性能更好特别是在并行流处理时。3. 多级排序的实战技巧真实项目中的排序需求往往像洋葱一样有多层。去年做HR系统时遇到个典型场景员工要先按部门字母顺序排同部门再按职级倒序职级相同的再按入职时间正序。用thenComparing组合起来就像搭积木employees.stream() .sorted(Comparator.comparing(Employee::getDepartment) .thenComparing(Employee::getLevel, Comparator.reverseOrder()) .thenComparing(Employee::getHireDate)) .forEach(System.out::println);这里有个容易踩的坑thenComparing连接的每个Comparator必须独立完整。有次我写成.thenComparing(Employee::getLevel.reversed())直接编译报错因为reversed()不是字段自带的属性。正确做法是像上面那样用Comparator.reverseOrder()或者用方法引用加比较器.thenComparing(Employee::getLevel, (a,b)- b.compareTo(a))。对于可能为null的字段可以用Comparator.nullsFirst()或Comparator.nullsLast()。比如处理用户列表时有些用户可能没填生日users.stream() .sorted(Comparator.comparing(User::getBirthday, Comparator.nullsLast(Comparator.naturalOrder())))这相当于把空值当作无穷大或无穷小处理避免出现NullPointerException。4. 性能优化与特殊场景当处理大数据量时排序可能成为性能瓶颈。有次我处理10万条日志数据发现sorted()操作耗时占整个流程的70%。后来通过测试发现几个优化点对于已经实现Comparable的对象直接使用sorted()比sorted(Comparator.comparing(...))更快因为少了一层包装在多级排序中把过滤条件高的字段放在前面能减少后续比较次数对于固定排序规则可以预编译Comparatorprivate static final ComparatorEmployee EMP_COMPARATOR Comparator.comparing(Employee::getDepartment) .thenComparingInt(Employee::getLevel);这样每次排序时就不用重复创建比较器实例了。在百万级数据测试中这种写法能提升约15%的性能。另一个特殊场景是对中文排序。直接用sorted()会按Unicode编码排可能不符合预期。可以用Collator实现本地化排序Collator zhCollator Collator.getInstance(Locale.CHINA); users.stream() .sorted(Comparator.comparing(User::getName, zhCollator))处理日期字符串时也要小心比如2021-9-1和2021-10-1按字符串排序会出错应该先转成LocalDate再比较Comparator.comparing(s - LocalDate.parse(s, DateTimeFormatter.ofPattern(yyyy-M-d)))5. 那些年我踩过的排序坑在实际项目中遇到过几个值得分享的案例。有次做订单导出功能要求按订单状态分组后每组内部按金额降序排。第一版代码写成orders.stream() .sorted(Comparator.comparing(Order::getStatus) .thenComparing(Order::getAmount).reversed())结果发现金额全是升序——原来reversed()作用的是整个Comparator链。正确写法应该是.thenComparing(Order::getAmount, Comparator.reverseOrder())还有个内存溢出的坑。有次对数据库查询结果直接做sorted().limit(100)结果当数据量很大时sorted()会先把所有数据加载到内存。后来改用数据库分页查询内存排序结合的方式解决。对于自定义复杂排序规则比如VIP用户优先然后按活跃度但黑名单用户永远在最后可以这样写ComparatorUser vipFirst (u1, u2) - { if(u1.isBlacklist() ! u2.isBlacklist()) { return u1.isBlacklist() ? 1 : -1; } if(u1.isVip() ! u2.isVip()) { return u1.isVip() ? -1 : 1; } return Integer.compare(u1.getActivity(), u2.getActivity()); };这种写法虽然不如链式调用优雅但在复杂业务规则下更灵活可控。

相关文章:

Java8 Stream sorted排序实战:从Comparator基础到多级排序进阶

1. 从零开始理解Stream sorted排序 第一次接触Java8的Stream sorted方法时,我盯着那段链式调用的代码看了足足十分钟。就像刚拿到新手机的老人,明明按键就在眼前,却不知道从哪下手。后来在实际项目中踩过几次坑才明白,sorted()本质…...

DataX 实战:从零构建跨库数据同步解决方案

1. 为什么选择DataX进行跨库数据同步 第一次接触DataX是在处理一个电商平台的订单数据迁移项目。当时需要将MySQL中的3000万条订单数据同步到阿里云的AnalyticDB进行分析,尝试了多种方案后,DataX的表现让我印象深刻。相比传统的SQL导出导入方式&#xff…...

Excel炒股党必备:手把手教你用Power Query免费获取并刷新股票历史数据

Excel炒股党必备:手把手教你用Power Query免费获取并刷新股票历史数据 在投资分析领域,数据更新速度往往决定着决策质量。对于习惯使用Excel的投资者来说,每次手动复制粘贴股票数据不仅效率低下,还容易出错。其实Excel内置的Power…...

管理SELinux安全性知识点问答

1.SELinux是如何保护资源的? SELinux给进程和文件指定了规则,严格按照规则限制文件和进程,默认拒绝所有未明确的操作来保护资源。 2.什么是强制访问控制(MAC)?它有什么特点? 强制访问控制是由系统统一强制决定进程/用户对文件/设备的访问权限。用户和…...

kotlin中一般用高介函数代替return

在 Kotlin 里完全可以不用 break ,而且日常开发基本都这么写。 我给你按场景列全,都是实际开发里最常用的替代方案,一看就会。集合高阶函数(最常用,直接替代 break) 找到第一个满足条件就停(等…...

AI编程革命:Codex如何重塑脚本开发效率

技术文章大纲:告别重复造轮子——利用Codex高效编写脚本核心价值与痛点分析重复性脚本开发的低效现状 人工编写脚本的常见问题:语法错误、逻辑冗余、调试耗时 Codex如何通过自然语言理解降低脚本开发门槛Codex基础能力解析自然语言到代码的转换机制 支持…...

Kelsey Hightower在KubeCon 2026:面对AI,人人都是初级工程师

Electrolux站点可靠性产品经理Kristina Kondrashevich清晰地记得Kelsey Hightower对她工作产生的深刻影响。"我们参加了KubeCon 2023,Kelsey Hightower在那次大会上做了一场关于开源项目的演讲,"Kondrashevich告诉The New Stack,&q…...

告别数据焦虑:用MedAugment给你的医学影像数据集‘打鸡血’(附Python实战代码)

告别数据焦虑:用MedAugment给你的医学影像数据集‘打鸡血’(附Python实战代码) 当你面对只有几十张标注好的医学影像数据时,是否感到无从下手?作为经历过这种困境的开发者,我清楚地记得第一次尝试用200张皮…...

Allegro PCB覆铜设计的10个高效技巧

1. 覆铜基础设置:从零开始的高效起点 刚接触Allegro PCB设计时,我最常犯的错误就是忽略覆铜的基础设置。很多人觉得覆铜就是随便画个形状填满铜皮,但实际工作中,合理的初始设置能节省50%以上的后期修改时间。在Allegro 16.6之后的…...

Sunshine游戏串流技术架构深度解析

Sunshine游戏串流技术架构深度解析 【免费下载链接】Sunshine Self-hosted game stream host for Moonlight. 项目地址: https://gitcode.com/GitHub_Trending/su/Sunshine Sunshine作为开源自托管游戏串流服务器,通过Moonlight协议实现低延迟跨设备游戏共享…...

生成式AI隐私影响评估(PIA)标准化模板(含12项强制审计指标+自动打分系统)

第一章:生成式AI应用数据隐私保护 2026奇点智能技术大会(https://ml-summit.org) 生成式AI在内容创作、代码生成与客户服务等场景中快速落地,但其对训练数据与用户输入的高度依赖,使敏感信息泄露、成员推断(membership inference…...

高效处理SDF文件:拆分与分子属性数据清理实战

1. SDF文件基础与化学信息学应用 SDF(Structure Data File)是化学信息学领域最常用的分子数据存储格式之一。这种纯文本格式最初由MDL公司开发,现已成为药物研发和分子建模中的通用标准。一个典型的SDF文件包含三个核心部分:分子结…...

[具身智能-380]:Habitat仿真平台概述以及如何利用该平台进行模型训练或算法调试?

📘 Habitat 仿真平台详解与训练/调试指南 Habitat 是由 Meta AI (FAIR) 开源的 3D 具身智能仿真平台,专注于室内视觉导航、多模态交互、具身感知与对话式 AI。它在学术界与工业界被广泛用于 Vision-and-Language Navigation (VLN)、ObjectGoal Navigati…...

【独立开发2】- Netunnel 内网穿透软件 - 你也在找无限制、便宜的吗?

设计初衷 总是找不到一款没有限制、便宜、操作简单的内网穿透软件。定价:0.5元/Gb ,最低一元。 https://github.com/aifuqiang02/netunnel 下载地址 , 访问不了github 的小伙伴, 可以加QQ群。找群主。 1、软件首页 (一睹为快&a…...

2026个人创业项目,0基础做门店WiFi商业变现

2026线下实体店流量红利依旧很大,很多人不知道,门店WiFi其实是一个非常适合个人起步的轻创业项目,不需要门店、不需要人脉、不需要营业执照,个人主体就能直接落地上线。 日常开店的餐饮、棋牌室、宾馆、便利店,几乎每…...

Golang colly爬虫框架如何用_Golang colly教程【进阶】

c.Visit()未触发OnHTML最常见原因是请求被目标站拦截导致403,因Colly默认UA易被拒绝;需设自定义UserAgent、加OnResponse打印状态码、处理重定向、传完整URL、用Limit()控并发、解压gzip、避开JS渲染页、选稳定选择器、用连接池channel安全存库。为什么 …...

安卓应用开发全流程实践与技术要点详解

引言 随着移动互联网的深入发展,安卓操作系统凭借其开放性和庞大的用户基数,在全球移动设备市场占据着举足轻重的地位。这催生了市场对高质量安卓应用和优秀安卓开发工程师的持续需求。作为一名安卓开发工程师,其职责远不止于编写代码,更涉及从需求理解、架构设计、编码实…...

新手小白创建SQL数据库,增删查改,升序,降序,不足之处望海涵

-- 创建数据表 CREATE TABLE users( id INT PRIMARY KEY, name VARCHAR(50) NOT NULL, id_card VARCHAR(18) NOT NULL, phone VARCHAR(11) NOT NULL, address TEXT, password VARCHAR(50) NOT NULL ); -- 添加数据 INSERT INTO school.users VALUES(1,gaga,622723190102214,176…...

告别Foxy导航‘幽灵错误’:手把手教你修改BT XML与源码,一劳永逸

根治ROS2 Foxy导航超时故障:从行为树配置到源码修改的终极指南 当你的ROS2 Foxy机器人突然在导航任务中"僵直",控制台不断刷出Action server failed while executing action callback: "send_goal failed"的错误时,这很可…...

康耐视相机与发那科机器人Ethernet I/P直连配置与视觉引导编程实战

1. 工业视觉与机器人协同的基础原理 在精密装配和分拣场景中,毫米级定位精度是刚需。康耐视相机就像机器人的"眼睛",通过Ethernet I/P协议将视觉数据实时传输给发那科机器人,形成闭环控制。这套系统最核心的优势在于硬件直连带来的…...

AI绘画商用翻车实录:为什么你买的“商用授权”根本无效?——深度拆解Stable Diffusion生态中7层隐性版权链

第一章:生成式AI应用版权合规指南 2026奇点智能技术大会(https://ml-summit.org) 生成式AI在内容创作、代码生成、设计辅助等场景中广泛应用,但其训练数据来源、输出内容权属及商业使用边界存在显著法律不确定性。开发者与企业需主动构建版权风险识别与…...

Spring Data MongoDB 最佳实践:如何构建高效数据访问层

在微服务、内容平台、物联网、日志系统和实时业务中,MongoDB 因其灵活的数据模型、优秀的水平扩展能力和较高的写入吞吐,被大量用于承载半结构化数据。对于 Java/Spring 技术栈来说,Spring Data MongoDB 是最常用的数据访问框架之一。它屏蔽了…...

MTKLogger存储空间总是不够用?教你调整‘Limit Log Size’并合理分配内部与SD卡存储

MTKLogger存储优化实战:精准控制日志大小与智能分配策略 每次测试进行到关键时刻,突然发现日志文件被自动覆盖,那种感觉就像马拉松终点前被强行拉回起点。作为深度依赖MTKLogger的测试工程师,我们都经历过存储空间不足导致的珍贵数…...

ESP32 IDF 无刷电机开环控制(完整工程+代码解析)

前言 本文基于 ESP-IDF 环境(推荐v6.0及以上),实现双无刷电机开环控制,包含完整工程创建、代码实现、IO管脚定义、接线指南及核心代码解析,适配ESP32核心板,新手可直接照搬工程,快速上手无刷电机…...

STM32F103C8T6连接ZH03B传感器,手把手教你做一个桌面PM2.5监测仪(附完整代码)

STM32F103C8T6与ZH03B传感器实战:打造高精度桌面PM2.5监测系统 最近工作室的空气质量总让我隐隐担忧,尤其是看到窗外雾蒙蒙的天空时。作为硬件爱好者,我决定用STM32F103C8T6和ZH03B激光粉尘传感器搭建一个实时监测装置。这个不到巴掌大的小盒…...

前端构建工具

前端构建工具的演进与核心价值 在当今快节奏的前端开发领域,构建工具已成为提升效率的关键。从早期的手动文件合并到如今的自动化流程,构建工具不仅简化了开发流程,还优化了代码性能。无论是个人项目还是企业级应用,选择合适的构…...

从交通拥堵到疾病预测:动态贝叶斯网络(DBN)在智慧城市中的3个落地场景与避坑指南

动态贝叶斯网络在智慧城市中的实战应用:从交通优化到疾病预警 引言:当城市开始"思考" 清晨7:30的早高峰,城市交通指挥中心的大屏上,红色拥堵路段正在以某种规律蔓延;同一时刻,疾控中心的监测系统…...

人工智能之数学基础:求解非线性约束

本文重点 在前面我们学习了两种方法,一种是内部法,另外一种是外部法,本文我们将学习一种新的方法,这种方法叫做乘子法。 乘子法 我们都听过拉格朗日函数,乘子法中,使用拉格朗日函数来代替f(x),所以此时f(x)为: 和外点法(内点法)一样,现在我们需要构建乘子罚函数:…...

XUnity自动翻译器:Unity游戏本地化的专业解决方案,5分钟实现高效汉化

XUnity自动翻译器:Unity游戏本地化的专业解决方案,5分钟实现高效汉化 【免费下载链接】XUnity.AutoTranslator 项目地址: https://gitcode.com/gh_mirrors/xu/XUnity.AutoTranslator 你是否曾因为语言障碍而错过优秀的Unity游戏?面对…...

关于十家信奥赛培训机构的公开信息整理

信奥赛(全国青少年信息学奥林匹克竞赛)近年来关注度持续上升。CSP-J/S认证的报名人数从2021年的8万余人增长至2024年的12万余人。以下整理了十家机构的公开信息,供参考。一、妙小程成立于2017年,是三七互娱旗下的教育品牌。课程体…...