【JAVA】jdk8 Stream 排序精通
背景
jdk8的stream流能方便的排序,但是每次都要查资料,非常不方便,不确定,所以这次直接弄懂,不再迷茫。
转载请注明来源,创作不易,请多多支持。
基础排序
stream流 大家应该都比较熟悉了,毕竟jdk8出来多久了,言简意赅的讲解下
stream 流提供的排序的方法其实就两个 :
一个是无参的 Stream<T> sorted();
一个是有参的 Stream<T> sorted(Comparator<? super T> comparator);
基础写法-无参
无参的排序很简单,按照默认排序,测试代码如下
@Data@Builderpublic static class User {private String username;private Integer age;}@Testpublic void StreamSort() {List<String> list1 = Lists.newArrayList("1", "2", "01", "02", "10", "11", "a");List<Integer> list2 = Lists.newArrayList(1, 2, 0, 3, 7);List<Boolean> list3 = Lists.newArrayList(true, false, true, false, false);List<User> list4 = Lists.newArrayList(User.builder().age(1).build(), User.builder().age(2).build(), User.builder().age(3).build());list1.stream().sorted().forEach(p -> System.out.print(p + " "));}
结论: 其实只要知道基础类型默认是怎么排序的就好了,对象类型本质还是指定到基础类型去,记住一个词 点名(从小到大)
1、字符串排序结果是 01 02 1 10 11 2 a,字符串比较很简单,就是字符依次比较,从小到大,第一位相同比第二位,依次排序。
2、数值类型排序结果 0 1 2 3 7,数值类型的都是一样的规律,从小到大 和 报数一样
3、布尔类型排序结果 false false false true true,可以类比 0是false 1是true 约定俗成 0为false 1为true, 依然是上升趋势
基础写法-有参
Comparator 是一个专门排序用的接口,只有一个核心方法 int compare(T o1, T o2);
严格的写法
一个匿名内部类实现排序的逻辑
list2.stream().sorted(new Comparator<Integer>() {@Overridepublic int compare(Integer p1, Integer p2) {return p1 - p2;}}).forEach(p -> System.out.print(p + " "));
jdk8可以用lambda表达式简写,非数值类型的需要自定义规则
(p1, p2) -> {return p1 - p2;}或者 进一步简写 (p1, p2) -> p1 - p2
常用便捷写法
大部分其实都是数值比较,除了非数值类型的需要自定义规则,
比如要实现默认的字符串的比较,可以自定义以下逻辑
list1.stream().sorted((p1, p2) -> {char[] chars1 = p1.toCharArray();char[] chars2 = p2.toCharArray();//取短数组长度for (int i = 0; i < Math.min(chars1.length, chars2.length); i++) {if (chars1[i] != chars2[i]) {return chars1[i] - chars2[i];}}return 0;}).forEach(p -> System.out.print(p + " "));
但是对于完全的数值比较,其实就按照默认的比较即可,给Stream 排序的值即可,特别是对象,只需要指定字段即可
list4.stream().sorted(Comparator.comparingInt(User::getAge)).forEach(p -> System.out.print(p + " "));
支持的数值方法有 comparingInt 、comparingLong、comparingDouble
基础写法-倒序
默认的排序是点名,从小到大,而有时候我们需要从大到小
除了自定义的排序方法,还有两种快捷的方式
1、在已完成的Comparator对象上再调用 reversed()方法,依然返回Comparator对象,不过是翻转后的,从小到大就变成了从大到小
2、对于对象,可以直接使用Comparator.comparing(User::getAge,Comparator.naturalOrder()), 这样的快捷构造来指定排序。
第一个就是排序的字段,第二个表示自然的排序naturalOrder(从小到大) 还是倒序reverseOrder (从大到小)
因为对象比基本类型占用空间大,如果排完序之后再翻转,而同时数组长度比较大,那么就可能有性能问题,所以可以直接指定排序的方向,避免不必要的浪费
这一点可以追溯一下,这里传入的第二个参数,其实就是排序方法的实例


NaturalOrderComparator 比较是 c1.compareTo(c2);
而 ReverseComparator 比较是 c2.compareTo(c1);
那么就比较明显了,他不是最终翻转而是比较过程中就已经取反了
因此对象排序,优先使用这样的方式去指定排序,特别是倒序的时候
高级排序
部分高级排序用法,程序设计的非常完善,有效好用
空值处理
Stream流处理不会处理空值,也就是Null,如果原始Stream流里面有空值,那么就直接会报空指针异常
如果可能有空值,一般需要过滤掉空值之后再排序,filter 返回布尔值,只保留返回true的数据
list4.stream().filter(p -> p.getAge() != null).sorted(Comparator.comparing(User::getAge, Comparator.reverseOrder())).forEach(p -> System.out.print(p + " "));
如果空值还需要处理(比如追加列表最后),一般用单独的数组存放
List<User> emptyList = list4.stream().filter(p -> p.getAge() == null).collect(Collectors.toList());List<User> resList = list4.stream().filter(p -> p.getAge() != null).sorted(Comparator.comparing(User::getAge, Comparator.reverseOrder())).collect(Collectors.toList());resList.addAll(emptyList);
多字段排序
可能单个字段排序不够,需要两次排序,甚至多次排序
其实也已经提供了现成的方法,thenComparing,同样返回的还是Comparator对象,那么可以继续追加
比如 我的原始数据是
List<User> list4 = Lists.newArrayList(User.builder().age(1).score(2).build(), User.builder().age(2).score(2).build(), User.builder().age(2).score(1).build(), User.builder().age(2).score(8).build(), User.builder().age(3).score(8).username("qiushi").build(), User.builder().age(3).score(5).build());
而排序方法是
.sorted(Comparator.comparing(User::getAge, Comparator.naturalOrder()).thenComparing(User::getScore, Comparator.naturalOrder()))
那么最终结果就是先 age 顺序,score顺序
User(username=null, age=1, score=2)
User(username=null, age=2, score=1)
User(username=null, age=2, score=2)
User(username=null, age=2, score=8)
User(username=null, age=3, score=5)
User(username=qiushi, age=3, score=8)
所以你明白了么? 先写到这里,后续再来补充,欢迎讨论指正,biu~
结论速记
1、默认是点名 从小到大排序,空值需自己处理
2、可以自定义排序方法,使用匿名函数(lambda表达式)快速实现,提供快速 转化数值的方法comparingInt 、comparingLong、comparingDouble
3、对象如果需要倒序,尽量指定排序方向,Comparator.comparing(User::getAge,Comparator.naturalOrder()),比最后来翻转reversed()的性能更好
4、支持多字段排序,使用thenComparing 方法依次指定
(日常偷图,侵删)

相关文章:
【JAVA】jdk8 Stream 排序精通
背景 jdk8的stream流能方便的排序,但是每次都要查资料,非常不方便,不确定,所以这次直接弄懂,不再迷茫。 转载请注明来源,创作不易,请多多支持。 基础排序 stream流 大家应该都比较熟悉了&…...
python的opencv操作记录12——Canny算子使用
文章目录Canny算子非极大值抑制非极大值抑制中的插值滞后阈值实际应用直接使用Canny算子使用膨胀先阈值分割Canny算子 上一篇说到,我在一个小项目里需要在一幅图像中提取一根试管里的两种液体的截面。为了达到这个目的使用传统图像里的区域分割技术,实际…...
Spark on hive Hive on spark
文章目录Spark on hive & Hive on sparkHive 架构与基本原理Spark on hiveHive on sparkSpark on hive & Hive on spark Hive 架构与基本原理 Hive 的核心部件主要是 User Interface(1)和 Driver(3)。而不论是元数据库&a…...
【MySQL】子查询
这里写自定义目录标题子查询1、子查询的基本使用2、 单行子查询2.1、单行比较查询2.2、HAVING 中的子查询2.3、CASE中的子查询3、多行子查询4、相关子查询5、EXISTS 与 NOT EXISTS关键字子查询 子查询指一个查询语句嵌套在另一个查询语句内部的查询,这个特性从MySQ…...
Day889.MySQL高可用 -MySQL实战
MySQL高可用 Hi,我是阿昌,今天学习记录的是关于MySQL高可用的内容。 正常情况下,只要主库执行更新生成的所有 binlog,都可以传到备库并被正确地执行,备库就能达到跟主库一致的状态,这就是最终一致性。但是…...
剑指 Offer 24. 反转链表
⭐简单说两句⭐ CSDN个人主页:后端小知识 🔎GZH:后端小知识 🎉欢迎关注🔎点赞👍收藏⭐️留言📝 题目: 剑指 Offer 24. 反转链表 ,我们今天还是来看一道easy的题目吧&…...
“黑铁时代”,地产人如何以客户视角加速房企数字化转型
本文从行业洞察、业务设计、数据建设以及实践探索四个部分详细阐述地产行业数字化的实践、思考和理解。点击文末“阅读原文”,观看完整版直播回放并下载演讲文档。一、洞察:房企经营思路的变化企业的转型都是围绕着业务经营变化进行的,房企数…...
零入门kubernetes网络实战-14->基于veth pair、namespace以及路由技术,实现跨主机命名空间之间的通信测试案例
《零入门kubernetes网络实战》视频专栏地址 https://www.ixigua.com/7193641905282875942 本篇文章视频地址(稍后上传) 本篇文章继续提供测试案例: 基于veth pair、namespace以及路由技术,实现跨主机命名空间之间的通信 1、网络拓扑如下 2、网络拓扑构建…...
【pytorch框架】对模型知识的基本了解
文章目录TensorBoard的使用1、TensorBoard启动:2、使用TensorBoard查看一张图片3、transforms的使用pytorch框架基础知识1 nn.module的使用2 nn.conv2d的使用3、池化(MaxPool2d)4 非线性激活5 线性层6 Sequential的使用7 损失函数与反向传播8 优化器9 对现有网络的使…...
SUP桨板电动气泵方案——鼎盛合方案
SUP桨板是现时最热门的水上运动之一,它的全称是Stand Up Paddle,简称SUP。这项运动近几年在我国三亚等地区风靡一时,在网上经常看到一些运动博主或者明星网红晒出冲浪视频,刺激又惊险。SUP桨板为充气式桨板,需要通过充…...
小白系列Vite-Vue3-TypeScript:011-登录界面搭建及动态路由配置
前面几篇文章我们介绍的都是ViteVue3TypeScript项目中环境相关的配置,接下来我们开始进入系统搭建部分。本篇我们来介绍登录界面搭建及动态路由配置,大家一起撸起来......搭建登录界面登陆接口api项目登陆接口是通过mockjs前端来模拟的模拟服务接口Login…...
C语言( 缓冲区和重定向)
一.缓冲输入,无缓存输入 while((chgetchar()) ! #) putchar(ch); 这里getchar(),putchar()每次只处理一个字符(这里只是知道就好了),而我们使用while循环,当读到#字符时停止 而看到输出例子,第一行我们输入…...
编程思想、方法论和架构的类型及应用
概要编程思想是指在编写代码时所采用的基本思维方式和方法论。分类编程思想编程思想为软件开发提供了思维范式和指导思路,例如面向对象思想、函数式编程思想等,它们帮助程序员更好地抽象问题、组织代码、提高代码复用性和可维护性,包括一下几…...
【OA办公】OA流程审批大揭秘,带你看遍所有基础流程
流程审批,是所有企业的OA办公系统重要组成部分,是任何OA办公系统都不可缺少的。比起传统的纸张传阅、签批的审批模式浪费了大量的时间和成本,因此越来越多的企业采用OA这种全新的、高效的、智能的审批模式。流程审批除了这些好处,…...
《零基础入门数据结构与算法》专栏介绍
目录 前言 第一部分:重点 第二部分:题库 第三部分:测试 第四部分:实验 第五部分:试卷 总结 前言 本专栏主要分为五个部分: ① 重要知识点详解 ② 近百道练习题解析 ③ 数据结构与算法章节测试 …...
测试开发之Django实战示例 第九章 扩展商店功能
第九章 扩展商店功能在上一章里,为电商站点集成了支付功能,然后可以生成PDF发票发送给用户。在本章,我们将为商店添加优惠码功能。此外,还会学习国际化和本地化的设置和建立一个推荐商品的系统。本章涵盖如下要点:建立…...
【Spring】一文带你吃透AOP面向切面编程技术(下篇)
个人主页: 几分醉意的CSDN博客_传送门 上节我们介绍了什么是AOP、Aspectj框架的前置通知Before传送门,这篇文章将继续详解Aspectj框架的其它注解。 文章目录💖Aspectj框架介绍✨JoinPoint通知方法的参数✨后置通知AfterReturning✨环绕通知Ar…...
【java】Spring Boot --40 个 Spring Boot 常用注解(建议收藏)
本文目录一、Spring Web MVC 注解Spring Web MVC 注解RequestMappingRequestBodyGetMappingPostMappingPutMappingDeleteMappingPatchMappingControllerAdviceResponseBodyExceptionHandlerResponseStatusPathVariableRequestParamControllerRestControllerModelAttributeCross…...
《游戏学习》| 微信对话模拟生成器源码分析
简介微信对话生成器,是一款在线微信聊天对话制作的工具,它可以设置苹果或安卓状态栏,包括手机电量、手机时间等,还可以设置不同用户的角色,然后发送文字、语音、红包、转账等多种好玩的功能,可谓是一款娱乐…...
剑指 Offer 10- I. 斐波那契数列[c语言]
目录题目思路代码结果该文章只是用于记录考研复试刷题题目 力扣斐波那契数列 写一个函数,输入 n ,求斐波那契(Fibonacci)数列的第 n 项(即 F(N))。斐波那契数列的定义如下: F(0) 0, F(1) 1 …...
Docker 离线安装指南
参考文章 1、确认操作系统类型及内核版本 Docker依赖于Linux内核的一些特性,不同版本的Docker对内核版本有不同要求。例如,Docker 17.06及之后的版本通常需要Linux内核3.10及以上版本,Docker17.09及更高版本对应Linux内核4.9.x及更高版本。…...
Zustand 状态管理库:极简而强大的解决方案
Zustand 是一个轻量级、快速和可扩展的状态管理库,特别适合 React 应用。它以简洁的 API 和高效的性能解决了 Redux 等状态管理方案中的繁琐问题。 核心优势对比 基本使用指南 1. 创建 Store // store.js import create from zustandconst useStore create((set)…...
无法与IP建立连接,未能下载VSCode服务器
如题,在远程连接服务器的时候突然遇到了这个提示。 查阅了一圈,发现是VSCode版本自动更新惹的祸!!! 在VSCode的帮助->关于这里发现前几天VSCode自动更新了,我的版本号变成了1.100.3 才导致了远程连接出…...
对WWDC 2025 Keynote 内容的预测
借助我们以往对苹果公司发展路径的深入研究经验,以及大语言模型的分析能力,我们系统梳理了多年来苹果 WWDC 主题演讲的规律。在 WWDC 2025 即将揭幕之际,我们让 ChatGPT 对今年的 Keynote 内容进行了一个初步预测,聊作存档。等到明…...
实现弹窗随键盘上移居中
实现弹窗随键盘上移的核心思路 在Android中,可以通过监听键盘的显示和隐藏事件,动态调整弹窗的位置。关键点在于获取键盘高度,并计算剩余屏幕空间以重新定位弹窗。 // 在Activity或Fragment中设置键盘监听 val rootView findViewById<V…...
[Java恶补day16] 238.除自身以外数组的乘积
给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O(n) 时间复杂度…...
RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程
本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在聚客AI学院。 本文全面剖析RNN核心原理,深入讲解梯度消失/爆炸问题,并通过LSTM/GRU结构实现解决方案,提供时间序列预测和文本生成…...
第一篇:Liunx环境下搭建PaddlePaddle 3.0基础环境(Liunx Centos8.5安装Python3.10+pip3.10)
第一篇:Liunx环境下搭建PaddlePaddle 3.0基础环境(Liunx Centos8.5安装Python3.10pip3.10) 一:前言二:安装编译依赖二:安装Python3.10三:安装PIP3.10四:安装Paddlepaddle基础框架4.1…...
【WebSocket】SpringBoot项目中使用WebSocket
1. 导入坐标 如果springboot父工程没有加入websocket的起步依赖,添加它的坐标的时候需要带上版本号。 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId> </dep…...
云安全与网络安全:核心区别与协同作用解析
在数字化转型的浪潮中,云安全与网络安全作为信息安全的两大支柱,常被混淆但本质不同。本文将从概念、责任分工、技术手段、威胁类型等维度深入解析两者的差异,并探讨它们的协同作用。 一、核心区别 定义与范围 网络安全:聚焦于保…...
