java8实战读书笔记:复合Lambda表达式

简介: java8实战读书笔记:复合Lambda表达式

本节将重点探讨复合Lambda表达式的使用。在阅读本篇之前建议您先阅读:


java8实战读书笔记:Lambda表达式语法与函数式编程接口,因为本篇是上篇的补充。


本书为博主在学习《java8实战》时的笔记。


image.png

f28d73b7457d91ca7b1b91e6d20bd93c.png

上面是JDK8中java.util.Comparator接口,相比jdk1.7增加了好多方法,也许你会觉得奇怪,为什么接口中还能定义方法,原因是JDK8中,可以为接口添加默认实现,使用default关键字定义。


我们可以这样定义一个比较器:


1Comparator< Apple>  c = Comparator.comparing(Apple::getWeight());

其等价为:

1Comparator< Apple>  c = Comparator.comparing(  (a) -> a.getWeight()  );


为什么可以这样写呢?因为Comparator定义了如下静态方法:

a518899017c781cbd770dde81ffb35fa.jpg


逆序


Comparator定义了一个静态方法,reversed,故我们不需要重新再定义一个比较器,我们可以这样就能实现逆序排序:


1List< Apple> apples = new ArrayList<>();
2apples.sort(  Comparator.comparing(Apple::getWeight()).reversed()  );


比较器链


如果要支持多重排序呢?例如先根据苹果的重量,如果重量相同就按照颜色排序,那如何来实现呢?

1apples.sort(  Comparator.comparing(Apple::getWeight()).thenComparing(  Apple::getColor()  ) );

之所以可以使用上述表达式,是因为Comparator定义了如下方法:

1default <U extends Comparable<? super U>> Comparator<T> thenComparing( Function<? super T, ? extends U> keyExtractor)
2{
3    return thenComparing(comparing(keyExtractor));
4}

温馨提示:Comparator中定义很多thenComparing重载方法,在具体使用过程中,可以先看看其函数声明。


image.png

提到谓词复合,我们就不得不提Predicate<T>函数式编程接口,其类图如下所示:

a6896aebb6d6d32662f2af0bb3e591dc.png

  • and:与
  • negate:非
  • or:或

温馨提示:and 和 or 是按照在表达式链中的位置,从左向右确定优先级的。因此, a.or(b).and(c) 可以看作 (a || b) && c 。


使用示例:从苹果列表中找出所有红色的,并且重量超过150的苹果:


1apples.filter((a -> "red".equals(a.getColor())).and(  a -> a.getWeight() > 150 ));
2a -> "red".equals(a.getColor())  是 (Apple a ) -> "red".equals(a.getColor())的简写。


image.png

函数复合,其对应的函数式编程接口为Function,其类图如下:

81cbe30bb952b0602a77282e521a6e7e.png

  • addThen


andThen 方法会返回一个函数,它先对输入应用一个给定函数,再对输出应用另一个函数。
例如:
Function f = x -> x + 1;
Function g = x -> x * 2;
Function h = f.andThen(g);
int result = h.apply(1);  // 其结果返回4,类似与数学公式  f(g(x))。


  • compose


先把给定的函数用作 compose 的参数里面给的那个函数,然后再把函数本身用于结果。与addThen的函数应用方向相反,同样举例说明如下:


Function f = x -> x + 1;
Function g = x -> x * 2;
Function h = f.compose(g);
int result = h.apply(1);  // 其结果返回3,类似与数学公式  g(f(x))。

相关文章
|
3月前
|
存储 人工智能 算法
从零掌握贪心算法Java版:LeetCode 10题实战解析(上)
在算法世界里,有一种思想如同生活中的"见好就收"——每次做出当前看来最优的选择,寄希望于通过局部最优达成全局最优。这种思想就是贪心算法,它以其简洁高效的特点,成为解决最优问题的利器。今天我们就来系统学习贪心算法的核心思想,并通过10道LeetCode经典题目实战演练,带你掌握这种"步步为营"的解题思维。
|
3月前
|
安全 Java 开发者
告别NullPointerException:Java Optional实战指南
告别NullPointerException:Java Optional实战指南
288 119
|
4月前
|
人工智能 Java API
Java AI智能体实战:使用LangChain4j构建能使用工具的AI助手
随着AI技术的发展,AI智能体(Agent)能够通过使用工具来执行复杂任务,从而大幅扩展其能力边界。本文介绍如何在Java中使用LangChain4j框架构建一个能够使用外部工具的AI智能体。我们将通过一个具体示例——一个能获取天气信息和执行数学计算的AI助手,详细讲解如何定义工具、创建智能体并处理执行流程。本文包含完整的代码示例和架构说明,帮助Java开发者快速上手AI智能体的开发。
1390 8
|
4月前
|
人工智能 Java API
Java与大模型集成实战:构建智能Java应用的新范式
随着大型语言模型(LLM)的API化,将其强大的自然语言处理能力集成到现有Java应用中已成为提升应用智能水平的关键路径。本文旨在为Java开发者提供一份实用的集成指南。我们将深入探讨如何使用Spring Boot 3框架,通过HTTP客户端与OpenAI GPT(或兼容API)进行高效、安全的交互。内容涵盖项目依赖配置、异步非阻塞的API调用、请求与响应的结构化处理、异常管理以及一些面向生产环境的最佳实践,并附带完整的代码示例,助您快速将AI能力融入Java生态。
678 12
|
4月前
|
Java 开发者
Java并发编程:CountDownLatch实战解析
Java并发编程:CountDownLatch实战解析
487 100
|
4月前
|
存储 前端开发 Java
【JAVA】Java 项目实战之 Java Web 在线商城项目开发实战指南
本文介绍基于Java Web的在线商城技术方案与实现,涵盖三层架构设计、MySQL数据库建模及核心功能开发。通过Spring MVC + MyBatis + Thymeleaf实现商品展示、购物车等模块,提供完整代码示例,助力掌握Java Web项目实战技能。(238字)
472 0
|
5月前
|
Java 编译器 API
Java Lambda表达式与函数式编程入门
Lambda表达式是Java 8引入的重要特性,简化了函数式编程的实现方式。它通过简洁的语法替代传统的匿名内部类,使代码更清晰、易读。本文深入讲解Lambda表达式的基本语法、函数式接口、方法引用等核心概念,并结合集合操作、线程处理、事件回调等实战案例,帮助开发者掌握现代Java编程技巧。同时,还解析了面试中高频出现的相关问题,助你深入理解其原理与应用场景。
|
3月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
216 1
|
3月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
237 1
|
4月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案