《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 

 

后记

 

《如梦令·迎雪夜归》

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

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

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

 

 


相关文章
|
6月前
|
安全 Java API
Java中的Lambda表达式:简洁与功能的结合
Java中的Lambda表达式:简洁与功能的结合
496 211
|
6月前
|
安全 Java
Java中的Switch表达式:更简洁的多路分支
Java中的Switch表达式:更简洁的多路分支
552 211
|
6月前
|
Java 编译器
Java 17 Switch表达式:更简洁、更强大的流程控制
Java 17 Switch表达式:更简洁、更强大的流程控制
753 111
|
6月前
|
Java 编译器 API
Java Lambda表达式与函数式编程入门
Lambda表达式是Java 8引入的重要特性,简化了函数式编程的实现方式。它通过简洁的语法替代传统的匿名内部类,使代码更清晰、易读。本文深入讲解Lambda表达式的基本语法、函数式接口、方法引用等核心概念,并结合集合操作、线程处理、事件回调等实战案例,帮助开发者掌握现代Java编程技巧。同时,还解析了面试中高频出现的相关问题,助你深入理解其原理与应用场景。
|
7月前
|
自然语言处理 Java Apache
在Java中将String字符串转换为算术表达式并计算
具体的实现逻辑需要填写在 `Tokenizer`和 `ExpressionParser`类中,这里只提供了大概的框架。在实际实现时 `Tokenizer`应该提供分词逻辑,把输入的字符串转换成Token序列。而 `ExpressionParser`应当通过递归下降的方式依次解析
408 14
|
8月前
|
SQL JSON 安全
Java 8 + 中 Lambda 表达式与 Stream API 的应用解析
摘要:本文介绍了Java 8+核心新特性,包括Lambda表达式与Stream API的集合操作(如过滤统计)、函数式接口的自定义实现、Optional类的空值安全处理、接口默认方法与静态方法的扩展能力,以及Java 9模块化系统的组件管理。每个特性均配有典型应用场景和代码示例,如使用Stream统计字符串长度、Optional处理Map取值、模块化项目的依赖声明等,帮助开发者掌握现代Java的高效编程范式。(150字)
161 1
|
7月前
|
设计模式 数据采集 Java
Java正则表达式的基础知识,进阶至熟练掌握。
通过大量的练习来熟悉它们的识别模式、如何设计模式来解决实际问题,才能够逐步达到熟练掌握。更多的是通过实践、编写代码和解决真实问题来完善技能。在这方面,没有快速的捷径,唯有刻意练习和长时间的代码实践。
137 0
|
消息中间件 算法 Java
2023金九银十Java高级工程师面试 1000 题+答案(全)
前言 春节过后招聘旺季就到了,不知道大家是否准备好了,面对金三银四的招聘旺季,如果没有精心准备那笔者认为那是对自己不负责任;就我们 Java 程序员来说,多数的公司总体上面试都是以自我介绍+项目介绍+项目细节/难点提问+基础知识点考核+算法题这个流程下来的。 有些公司可能还会问几个实际的场景类的问题,这个环节阿里是必问的,这种问题通常是没有正确答案的,就看个人的理解,个人的积累了。剩下的就没啥了,都是换汤不换药,聊项目就看你自己对你自己的项目是否理解的透彻,比如经常问你你为什么选择这个技术,为什么这么处理之类的,常考的基础的知识点就那么多,最后算法就是靠刷题。 今天为大家分享的就是当前
|
XML JavaScript Java
华为Java工程师面试题
华为Java工程师面试题
144 0