jdk8 sorted 排序值comparing 和thenComparing 偶遇问题解析

2024-06-06 16:18

本文主要是介绍jdk8 sorted 排序值comparing 和thenComparing 偶遇问题解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在java8之前,排序有两种方法:

1.定义比较器类 继承Comparator接口,然后重写compare(a,b) 方法,返回的结果是int类型 

0  代表 a==b

1 代表 a>b

-1 代表 a<b

2.定义比较器对象

Comparator<Person> c1 = new Comparator<Person>() {@Overridepublic int compare(Person o1, Person o2) {return o1.getAge() - o2.getAge();}};Collections.sort(personList, c1);

好了,前面的排序就讲到这了,不是我要讲解的重点。

在jdk8中,排序可以使用stream中的sorted。

 personList.stream().sorted(Comparator.comparing((Person::getAge)).collect(Collectors.toList()); //按照年龄有小到大排序
personList.stream().sorted(Comparator.comparing((Person::getAge).thenComparing(Person::getId())).collect(Collectors.toList()) //先按年龄从小到大排序,年龄相同再按id从小到大排序

再来看看comparing和thenComparing的源码

public static <T, U extends Comparable<? super U>> Comparator<T> comparing(Function<? super T, ? extends U> keyExtractor)
{Objects.requireNonNull(keyExtractor);return (Comparator<T> & Serializable)(c1, c2) -> keyExtractor.apply(c1).compareTo(keyExtractor.apply(c2));
}
default <U extends Comparable<? super U>> Comparator<T> thenComparing(Function<? super T, ? extends U> keyExtractor)
{return thenComparing(comparing(keyExtractor));
}

default  就不说了,是一个jdk8的新特性,不明白的请自行google。

初次看这个两个源码,发现他们传入的参数类型是一模一样的,但是请注意返回类型可不一样,为什么这里要突显出来。还是请看代码:

personList.stream().sorted(Comparator.comparing(( p) -> -p.getCars().size()).thenComparing(Person::getId)).collect(Collectors.toList());

代码在p.getCars.size() 处报错,cannot resolve method "getCars()"。这里一定有人会纠结为什么我写的也是函数式编程,按道理没问题呀!但是编辑器就是不认识p.getCars()。

正解:在使用comparing的时候p没有指定类型,比较的时候就不知道是哪一种类型,自然也找不到这个类的方法。代码在编译的时候不通过,idea语法检查也不通过。如下代码就行,指定了p的类型Person。

personList.stream().sorted(Comparator.comparing((Person p) -> -p.getCars().size()).thenComparing(Person::getId)).collect(Collectors.toList());

当然如果将comparing里面的代码分开来写,看的会更加清楚,特别指定了类型是Person。

Function<Person, Integer> function = p -> -p.getCars().size();
personList = personList.stream().sorted(Comparator.comparing(function).thenComparing(Person::getId)).collect(Collectors.toList());

完整代码如下 :

public class Car {private String carName;public String getCarName() {return carName;}public void setCarName(String carName) {this.carName = carName;}@Override
    public String toString() {return "Car{" +"carName='" + carName + '\'' +'}';}
}
public class Person {private int id;private String name;private List<Car> cars;public List<Car> getCars() {return cars;}public void setCars(List<Car> cars) {this.cars = cars;}public Person(int id, String name) {this.id = id;this.name = name;}Person() {}public int getId() {return id;}public void setId(int id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}public static void main(String[] args) {Person p1 = new Person(9, "aa");p1.setCars(new ArrayList<>());Car car10 = new Car();car10.setCarName("car10");Car car11 = new Car();car11.setCarName("car11");p1.getCars().add(car10);p1.getCars().add(car11);Person p2 = new Person(3, "aa");p2.setCars(new ArrayList<>());Car car20 = new Car();car20.setCarName("car20");Car car21 = new Car();car21.setCarName("car21");Car car23 = new Car();car23.setCarName("car23");p2.getCars().add(car20);p2.getCars().add(car21);p2.getCars().add(car23);List<Person> personList = new ArrayList<>();personList.add(p1);personList.add(p2);Function<Person, Integer> function = p -> -p.getCars().size();personList = personList.stream().sorted(Comparator.comparing(function).thenComparing(Person::getId)).collect(Collectors.toList());System.out.println(personList);}@Override
    public String toString() {return "Person{" +"id=" + id +", name='" + name + '\'' +", cars=" + cars +'}';}
}
好了,今天就分享到这里,不知大家有没有注意到,-p.getCars.size() 可以按照Person里面Car的数量逆序排序。里面的东西还有很多,待你发现哦~

这篇关于jdk8 sorted 排序值comparing 和thenComparing 偶遇问题解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1036604

相关文章

使用Jackson进行JSON生成与解析的新手指南

《使用Jackson进行JSON生成与解析的新手指南》这篇文章主要为大家详细介绍了如何使用Jackson进行JSON生成与解析处理,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 核心依赖2. 基础用法2.1 对象转 jsON(序列化)2.2 JSON 转对象(反序列化)3.

Springboot @Autowired和@Resource的区别解析

《Springboot@Autowired和@Resource的区别解析》@Resource是JDK提供的注解,只是Spring在实现上提供了这个注解的功能支持,本文给大家介绍Springboot@... 目录【一】定义【1】@Autowired【2】@Resource【二】区别【1】包含的属性不同【2】@

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

SpringCloud动态配置注解@RefreshScope与@Component的深度解析

《SpringCloud动态配置注解@RefreshScope与@Component的深度解析》在现代微服务架构中,动态配置管理是一个关键需求,本文将为大家介绍SpringCloud中相关的注解@Re... 目录引言1. @RefreshScope 的作用与原理1.1 什么是 @RefreshScope1.

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

Java的IO模型、Netty原理解析

《Java的IO模型、Netty原理解析》Java的I/O是以流的方式进行数据输入输出的,Java的类库涉及很多领域的IO内容:标准的输入输出,文件的操作、网络上的数据传输流、字符串流、对象流等,这篇... 目录1.什么是IO2.同步与异步、阻塞与非阻塞3.三种IO模型BIO(blocking I/O)NI

SpringBoot启动报错的11个高频问题排查与解决终极指南

《SpringBoot启动报错的11个高频问题排查与解决终极指南》这篇文章主要为大家详细介绍了SpringBoot启动报错的11个高频问题的排查与解决,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一... 目录1. 依赖冲突:NoSuchMethodError 的终极解法2. Bean注入失败:No qu

Python 中的异步与同步深度解析(实践记录)

《Python中的异步与同步深度解析(实践记录)》在Python编程世界里,异步和同步的概念是理解程序执行流程和性能优化的关键,这篇文章将带你深入了解它们的差异,以及阻塞和非阻塞的特性,同时通过实际... 目录python中的异步与同步:深度解析与实践异步与同步的定义异步同步阻塞与非阻塞的概念阻塞非阻塞同步

MySQL新增字段后Java实体未更新的潜在问题与解决方案

《MySQL新增字段后Java实体未更新的潜在问题与解决方案》在Java+MySQL的开发中,我们通常使用ORM框架来映射数据库表与Java对象,但有时候,数据库表结构变更(如新增字段)后,开发人员可... 目录引言1. 问题背景:数据库与 Java 实体不同步1.1 常见场景1.2 示例代码2. 不同操作

如何解决mysql出现Incorrect string value for column ‘表项‘ at row 1错误问题

《如何解决mysql出现Incorrectstringvalueforcolumn‘表项‘atrow1错误问题》:本文主要介绍如何解决mysql出现Incorrectstringv... 目录mysql出现Incorrect string value for column ‘表项‘ at row 1错误报错