《Java工程师必读手册》——Java极客主义系列——最完美的Lambda表达式只有一行(7)

简介: 《Java工程师必读手册》——Java极客主义系列——最完美的Lambda表达式只有一行(7)

接上篇:https://developer.aliyun.com/article/1226751?spm=a2c6h.12873581.technical-group.dArticle1226751.524d5a5cxQxTxO


八、 Lambda表达式使用经验

 

下面,将介绍函数式编程和Lambda表达式的一些使用经验。

 

1. 规避传入可变参数

 

由于Lambda表达式只接受final不可变参数,或没有final修饰的无修改参数。如果希望传入的参数可修改,可以利用对象(容器对象不可变,但容器内容可变)解决。

 

image.png

 

如上所示,利用AtomicInteger解决了序号递增的问题。

 

2. 避免自定义函数式接口

 

为了减少代码中自定义函数式接口,可以利用Java提供的4个基本函数式接口:

 

Supplier提供者:没有输入参数,返回一个T类型的值;

Consumer(消费者):接收一个T类型的参数,并且没有返回值;

FunctionR>(函数):接收一个T类型的参数,返回一个R类型的值;

Predicate(断言):接收一个T类型的参数,返回一个boolean类型的值。

 

在包“java.util.function”中,Java提供了大量的函数式接口,大家可以实际需求选择使用。

 

原始代码

 

image.png

 

 

优化代码

 

image.png

 

3. 解决延迟调用问题

 

尽量使用方法引用传递代替值传递,可以让方法在真正使用时才调用,避免一些不必要的方法计算,我们称之为延迟调用

 

在下面代码中,无论“localValue”是否为空,都将执行“getRemoteValue”方法获取值。

 

image.png

 

那么,我们可以利用“Optional”的“orElseGet”方法实现延迟调用:

 

image.png

 

4. 解决初始化计算问题

 

根据模板工厂设计模式,需要抽象出一个或多个方法,并实现一个固定的算法流程。如果在算法流程中,循环地调用某一个方法,并且在这个方法中都要计算同一套参数,就存在一个初始化计算问题——应该提前初始化这些参数。

 

如果这些参数固定(每个实现类都用这套参数),我们可以抽象出一个初始化参数的虚方法,利用这个方法计算得出参数值,然后再把这些参数传入对应的方法。但是,如果这些参数不固定(根据实现类不同而不同),就不能采用这个方法了。

 

这里,作者给出一个基于Lambda表达式(函数式编程)的解决方案。

 

image.png

 

如上所示首先计算出参数k1、k2值,再返回一个基于这些参数实现的方法引用。主流程使用这个方法引用判断用户是否有效,从而避免了每次调用对参数k1、k2值的计算。

 

5. 解决类型推断冲突问题

 

有时候,重载的两个方法(方法名完全一致),为了区分是否有返回值,需要定义两个不同的参数类型。如果,这两个参数类型是函数式接口,且这两个函数式接口定义完全一致,在使用Lambda表达式时,就会出现类型推断冲突的问题。

 

image.png

 

通常,我们会如下编写Lambda表达式:

 

image.png

 

但是,这个时候编辑器就会报出以下错误:

 

image.png

 

那么,我们需要加上类型强制转换:

 

image.png

 

但是,SonarLint还会出现以下警告:

 

image.png

 

最后,给出最完美的解决方案:

 

image.png 

 

6. 简化常用的设计模式

 

在软件开发过程中,软件设计模式无处不在。通过函数式编程,减少了不必要接口声明及类实现,能够让设计模式的运用更灵活。

 

比如下面的代码中,明显有很多的通用代码。如果按照模板方法模式(Template Method Pattern)进行封装实现,将会编写大量的模式代码。

 

image.png

 

如果通过函数式编程和Lambda表达式实现,代码量将会降到最低。

 

image.png

 

7. 优化多表达式条件语句

 

有时候,我们会判断很多条件,需求用&&(或||)连接多个条件表达式。

 

image.png

 

通过SonarLint插件扫描,会存在2个复杂度问题:

 

Expressions should not be too complex(表达式不能太复杂);

Methods should not be too complex(方法不能太复杂)。

 

这里,可以采用Lambda表达式进行优化,降低代码复杂度并避免上述问题。

 

image.png 

 

后记

 

《如梦令·迎雪夜归》

冬日寒风砭骨,皓月冷光铺路。

一骑夜归迟,迎雪下车徒步。

何苦?何苦?即兴作诗吟赋!

 

 


相关文章
|
3天前
|
SQL Rust Java
怎么理解Java中的lambda表达式
Lambda表达式是JDK8引入的新语法,用于简化匿名内部类的代码写法。其格式为`(参数列表) -> { 方法体 }`,适用于函数式接口(仅含一个抽象方法的接口)。通过Lambda表达式,代码更简洁灵活,提升Java的表达能力。
|
8天前
|
SQL IDE 算法
《从头开始学java,一天一个知识点》之:运算符与表达式:算术、比较和逻辑运算
**你是否也经历过这些崩溃瞬间?** - 看了三天教程,连`i++`和`++i`的区别都说不清 - 面试时被追问"`a==b`和`equals()`的区别",大脑突然空白 - 写出的代码总是莫名报NPE,却不知道问题出在哪个运算符 这个系列为你打造Java「速效救心丸」,每天1分钟,地铁通勤、午休间隙即可完成学习。直击高频考点和实际开发中的「坑位」,拒绝冗长概念,每篇都有可运行的代码示例。明日预告:《控制流程:if-else条件语句实战》。
26 6
|
1月前
|
机器学习/深度学习 Java PyTorch
Java工程师如何理解张量?
刚接触AI和PyTorch,理解“张量(Tensor)”是入门关键。张量可类比为Java中的多维数组,但更强大,尤其在AI领域支持GPU加速、自动求导等特性。它不仅能高效存储数据,还能进行复杂运算,是深度学习的核心数据结构。掌握张量的维度、数据类型及GPU加速特性,对学习PyTorch至关重要。
52 3
|
3月前
|
Java API 开发者
Java中的Lambda表达式与Stream API的协同作用
在本文中,我们将探讨Java 8引入的Lambda表达式和Stream API如何改变我们处理集合和数组的方式。Lambda表达式提供了一种简洁的方法来表达代码块,而Stream API则允许我们对数据流进行高级操作,如过滤、映射和归约。通过结合使用这两种技术,我们可以以声明式的方式编写更简洁、更易于理解和维护的代码。本文将介绍Lambda表达式和Stream API的基本概念,并通过示例展示它们在实际项目中的应用。
|
3月前
|
安全 Java API
Java中的Lambda表达式:简化代码的现代魔法
在Java 8的发布中,Lambda表达式的引入无疑是一场编程范式的革命。它不仅让代码变得更加简洁,还使得函数式编程在Java中成为可能。本文将深入探讨Lambda表达式如何改变我们编写和维护Java代码的方式,以及它是如何提升我们编码效率的。
|
4月前
|
安全 Java API
Java中的Lambda表达式与Stream API的高效结合####
探索Java编程中Lambda表达式与Stream API如何携手并进,提升数据处理效率,实现代码简洁性与功能性的双重飞跃。 ####
51 0
|
设计模式 搜索推荐 Java
Java 极客技术 2019 年高考语文卷
Java 极客技术 2019 年高考语文卷
|
16天前
|
存储 监控 Java
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
142 60
【Java并发】【线程池】带你从0-1入门线程池
|
5天前
|
存储 网络协议 安全
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
55 23
|
12天前
|
Java 调度
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
当我们创建一个`ThreadPoolExecutor`的时候,你是否会好奇🤔,它到底发生了什么?比如:我传的拒绝策略、线程工厂是啥时候被使用的? 核心线程数是个啥?最大线程数和它又有什么关系?线程池,它是怎么调度,我们传入的线程?...不要着急,小手手点上关注、点赞、收藏。主播马上从源码的角度带你们探索神秘线程池的世界...
81 0
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码

热门文章

最新文章