1、概述

在这篇教程里,我们将要去了解下即将到来的JDK 8(译注,现在JDK 8已经发布了)中的Lambda表达式——特别是怎样使用它来编写Comparator和对集合(Collection)进行排序。

首先,让我们先定义一个简单的实体类:

public class Human {

private String name;

private int age;

public Human() {

super();

}

public Human(final String name, final int age) {

super();

this.name = name;

this.age = age;

}

// standard getters and setters

}

2、不使用Lambda表达式的基本排序

在Java 8之前,对集合进行排序要为Comparator创建一个匿名内部类用来排序:

new Comparator() {

@Override

public int compare(Human h1, Human h2) {

return h1.getName().compareTo(h2.getName());

}

}

简单地用它来对Human实体列表进行排序:

@Test

public void givenPreLambda_whenSortingEntitiesByName_thenCorrectlySorted() {

List humans = Lists.newArrayList(new Human("Sarah", 10), new Human("Jack", 12));

Collections.sort(humans, new Comparator() {

@Override

public int compare(Human h1, Human h2) {

return h1.getName().compareTo(h2.getName());

}

});

Assert.assertThat(humans.get(0), equalTo(new Human("Jack", 12)));

}

3、使用Lambda表达式的基本排序

根据Lambda表达式的介绍,我们现在可以不使用匿名内部类,只使用简单实用的语义就可以得到相同的结果。

(final Human h1, final Human h2) -> h1.getName().compareTo(h2.getName());

类似地,我们现在可以像之前那样来测试它的行为

@Test

public void whenSortingEntitiesByName_thenCorrectlySorted() {

List humans = Lists.newArrayList(new Human("Sarah", 10), new Human("Jack", 12));

humans.sort((Human h1, Human h2) ->

h1.getName().compareTo(h2.getName()));

Assert.assertThat(humans.get(0), equalTo(new Human("Jack", 12)));

}

注意:我们同样使用新的sort API,这个API在Java 8里被添加到java.util.List ——而不是旧的Collections.sort API。

4、没有类型定义( Type Definitions)的基本排序

我们通过不指定类型定义来进一步简化表达式 ——编译器自己可以进行类型判断:

(h1, h2) -> h1.getName().compareTo(h2.getName())

测试仍然很相似:

@Test

public void givenLambdaShortForm_whenSortingEntitiesByName_thenCorrectlySorted() {

List humans = Lists.newArrayList(new Human("Sarah", 10), new Human("Jack", 12));

humans.sort((h1, h2) -> h1.getName().compareTo(h2.getName()));

Assert.assertThat(humans.get(0), equalTo(new Human("Jack", 12)));

}

5、使用静态方法的引用来排序

下面我们将要使用带有静态方法引用的Lambda表达式去进行排序。

首先,我们要定义compareByNameThenAge方法 ——这个方法拥有与Comparator对象里的compareTo方法完全相同的签名:

public static int compareByNameThenAge(Human lhs, Human rhs) {

if (lhs.name.equals(rhs.name)) {

return lhs.age - rhs.age;

} else {

return lhs.name.compareTo(rhs.name);

}

}

现在,我们要使用这个引用去调用humans.sort方法:

humans.sort(Human::compareByNameThenAge);

最终结果是一个使用静态方法作为Comparator的有效的排序集合:

@Test

public void givenMethodDefinition_whenSortingEntitiesByNameThenAge_thenCorrectlySorted() {

List humans = Lists.newArrayList(new Human("Sarah", 10), new Human("Jack", 12));

humans.sort(Human::compareByNameThenAge);

Assert.assertThat(humans.get(0), equalTo(new Human("Jack", 12)));

}

6、提取Comparator进行排序

我们可以通过使用实例方法的引用和Comparator.comparing方法来避免定义比较逻辑——它会提取和创建一个基于那个函数的Comparable。

我们准备使用getName() getter方法去建造Lambda表达式并通过name对列表进行排序:

@Test

public void givenInstanceMethod_whenSortingEntitiesByNameThenAge_thenCorrectlySorted() {

List humans = Lists.newArrayList(new Human("Sarah", 10), new Human("Jack", 12));

Collections.sort(humans, Comparator.comparing(Human::getName));

Assert.assertThat(humans.get(0), equalTo(new Human("Jack", 12)));

}

7、反转排序

JDK 8同样提供了一个有用的方法用来反转Comparator(reverse Comparator)——我们可以快速地利用它来反转我们的排序:

@Test

public void whenSortingEntitiesByNameReversed_thenCorrectlySorted() {

List humans = Lists.newArrayList(

new Human("Sarah", 10), new Human("Jack", 12));

Comparator comparator = (h1, h2) -> h1.getName().compareTo(h2.getName());

humans.sort(comparator.reversed());

Assert.assertThat(humans.get(0), equalTo(new Human("Sarah", 10)));

}

8、多条件排序

比较操作的Lambda表达式不一定都是这么简单的——我们同样可以编写更复杂的表达式,比如先根据name后根据age来对实体进行排序:

@Test

public void whenSortingEntitiesByNameThenAge_thenCorrectlySorted() {

List humans = Lists.newArrayList(

new Human("Sarah", 12), new Human("Sarah", 10), new Human("Zack", 12));

humans.sort((lhs, rhs) -> {

if (lhs.getName().equals(rhs.getName())) {

return lhs.getAge() - rhs.getAge();

} else {

return lhs.getName().compareTo(rhs.getName());

}

});

Assert.assertThat(humans.get(0), equalTo(new Human("Sarah", 10)));

}

9、多条件组合排序

同样的比较逻辑——先根据name进行排序其次是age,同样可以通过Comparator新的组合支持来实现。

从JDK 8开始,我们现在可以把多个Comparator链在一起(chain together)去建造更复杂的比较逻辑:

@Test

public void givenComposition_whenSortingEntitiesByNameThenAge_thenCorrectlySorted() {

List humans = Lists.newArrayList(

new Human("Sarah", 12), new Human("Sarah", 10), new Human("Zack", 12));

humans.sort(Comparator.comparing(Human::getName).thenComparing(Human::getAge));

Assert.assertThat(humans.get(0), equalTo(new Human("Sarah", 10)));

}

10、总结

这篇文章举例说明了多种令人兴奋的方法:使用Java 8 Lambda表达式对列表进行排序——正确使用过去的语法糖和真正、强大实用的语义。

所有这些例子的实现和代码片段都可以在我的github项目上获取到——这是一个基于Eclipse的项目,所以它应该很容易被导入和运行。

java lambda sorted_Java8:Lambda表达式增强版Comparator和排序相关推荐

  1. Java8:Lambda表达式增强版Comparator和排序

    1.概述 在这篇教程里,我们将要去了解下即将到来的JDK 8(译注,现在JDK 8已经发布了)中的Lambda表达式--特别是怎样使用它来编写Comparator和对集合(Collection)进行排 ...

  2. java用lambda函数排序,Java8:使用Lambda表达式增强版Comparator排序

    学习路上的自我记录-------路好长,就问你慌不慌,大声港,不慌.----jstarseven. 实体类: package com.server.model; /** * Created by js ...

  3. Java8 Collections.sort()及Arrays.sort()中Lambda表达式及增强版Comparator的使用

    摘要:本文主要介绍Java8 中Arrays.sort()及Collections.sort()中Lambda表达式及增强版Comparator的使用. 不废话直接上代码 import com.goo ...

  4. comparator接口_8000字长文让你彻底了解 Java 8 的 Lambda、函数式接口、Stream 用法和原理

    我是风筝,公众号「古时的风筝」.一个兼具深度与广度的程序员鼓励师,一个本打算写诗却写起了代码的田园码农! 文章会收录在 JavaNewBee 中,更有 Java 后端知识图谱,从小白到大牛要走的路都在 ...

  5. Java 8 新增lambda表达式(-)

    2019独角兽企业重金招聘Python工程师标准>>> Java 8 刚于几周前发布,日期是2014年3月18日,这次开创性的发布在Java社区引发了不少讨论,并让大家感到激动.特性 ...

  6. 十三、Java高级特性 Lambda表达式 | 接口组成更新 | 方法引用 | 函数式接口

    文章目录 十三.Java高级特性 1.Lambda表达式 1.1体验Lambda表达式[理解] 1.2Lambda表达式的标准格式[理解] 1.3Lambda表达式练习1[应用] 1.4Lambda表 ...

  7. java基础之lambda表达式

    java基础之lambda表达式 1 什么是lambda表达式 lambda表达式是一个匿名函数,允许将一个函数作为另外一个函数的参数,将函数作为参数传递(可理解为一段传递的代码). 2 为什么要用l ...

  8. 如何开始使用Java中的Lambda表达式

    by Luis Santiago 路易斯·圣地亚哥(Luis Santiago) 如何开始使用Java中的Lambda表达式 (How to start working with Lambda Exp ...

  9. 如何在Java中使用Lambda表达式

    如何在Java中使用Lambda表达式 Lambda表达式是Java 8新增的一个非常强大的特性.它使得函数式编程在Java中变得更加容易和直观. Lambda表达式的基础知识 在Java中,Lamb ...

最新文章

  1. Hibernate初探
  2. mysql 搜索标题中字符串_如何在MySQL表中搜索特定字符串?
  3. python中使用for循环,while循环,一条命令打印99乘法表
  4. go get 指定版本_没有VPS搭建govanityurls服务?别急!你依然可以自定义Go包导入路径...
  5. (转)深入理解最强桌面地图控件GMAP.NET --- 原理篇
  6. MySQL编程技巧_MySQL学习笔记---MySQL开发技巧
  7. python中的super使用方法[探索4]
  8. android系统密码文件夹,安卓手机如何隐藏(加密)文件夹(软件) 安卓手机隐藏(加密)文件夹(软件)的方法...
  9. Mac环境下安装Pod
  10. Android自定义控件之TextView
  11. 标准ASCII编码表
  12. python语言判断中国节假日(pip install chinesecalendar)
  13. 每周一看:16份文档资料,程序员软硬实力全概览,总有一个适合你
  14. JSR303校验的简单使用以及自定义校验规则的代码编写
  15. web前端html5+css3学习笔记(2)
  16. Linux系统使用userdel命令删除用户
  17. Ubuntu18.04  网卡配置
  18. nginx服务器 java项目,服务器使用Nginx如何部署Springboot项目
  19. Java实现 LeetCode 187 重复的DNA序列
  20. 采用计算机模拟,计算机模拟试卷4

热门文章

  1. matlab 粘连分割代码,【求助】图像中粘连字符切分代码,求注释!!!
  2. linux误删文件咋恢复,恢复Linux误删文件
  3. java高级教程_Java高级教程02
  4. mysql 读取properties_JDBC中使用Properties读取配置文件有什么用
  5. JVM005_synchronized、同步指令、管程、MESA
  6. 一篇博客读懂设计模式之---动态代理与反射
  7. php中统计记录条数,使用GROUP BY的时候如何统计记录条数 COUNT(*) DISTINCT
  8. c++ int自动转换成无符号变量产生的问题
  9. sql相同顺序法和一次封锁法_数学专题 | Ep01 隔板法的妙用
  10. matlab 参数识别,[转载]自编最小二乘法的Matlab参数辨识程序(含实例)