Java 8 In Action Lambda
引言
在优化代码的过程中,通过使用Java 8 lambda
表达式,减少代码量,提高代码可读性。
lambda 表达式简介
Lambda 表达式可以理解为一种匿名函数,它没有名称,但有参数列表、函数体、返回类型,并且可能还有一个可能抛出的异常列表。
Lambda 表达式可以作为参数传递给方法,也可以存储在变量中。
与匿名内部类相比,Lambda 表达式更加简洁。
从一个例子入手:写一个比较苹果的重量的逻辑:
传统方式:
1
2
3
4
5Comparator<Apple> byWeight = new Comparator<Apple>() {
public int compare(Apple a1, Apple a2){
return a1.getWeight().compareTo(a2.getWeight());
}
};使用 lambda 表达式:
1
Comparator<Apple> byWeight = (Apple a1, Apple a2) -> a1.getWeight().compareTo(a2.getWeight());
必须承认,代码看起来变得更清晰了。
A lambda expression is composed of parameters, an arrow, and a body.
- 参数列表:在这个例子中,它反映了
Comparator
的compare
方法的参数(两个Apple
对象)。 - 箭头:箭头
->
将参数列表与lambda
的函数体分隔开。 Lambda
的函数体:使用它们的重量比较两个Apple
对象。这个表达式被认为是lambda
的返回值。
Lambda 的基本语法
1 | (parameters) -> expression // 表达式风格 |
简单的示例
1 | // 一个布尔表达式 |
函数式接口
在Java 8
的java.util.function
包中引入了一系列的函数式接口,如:Predicate
,Consumer
,和 Function
等。
函数式接口是一种特殊的接口,它只定义了一个抽象方法,这样的接口可以用来表示Lambda
表达式。
函数式接口特点:
- 函数式接口可以有一个或多个默认方法(
default methods
),这些默认方法可以有实现。 - 函数式接口可以有多个静态方法(
static methods
),这些静态方法也可以有实现。 - 函数式接口可以有多个从父接口继承的抽象方法,但这些方法必须都是唯一的,也就是说,函数式接口里只能有一个抽象方法。
- 函数式接口可以用
@FunctionalInterface
注解来标记,但这不是必须的。这个注解的作用是帮助编译器检查接口是否符合函数式接口的定义。
For a Example
Predicate<T>
:接受一个泛型T
的对象,并返回一个布尔值。常用于条件判断。1
2Predicate<String> predicate = s -> s.isEmpty();
boolean result = predicate.test("");Consumer<T>
:接受一个泛型T
的对象,并执行一些操作,但没有返回值。常用于数据消费。1
2Consumer<String> consumer = System.out::println;
consumer.accept("Hello World");Function<T, R>
:接受一个泛型T
的对象,并返回一个泛型R
的对象。常用于转换数据。1
2Function<Integer, String> function = String::valueOf;
String apply = function.apply(123);Supplier<T>
:提供一个泛型T
的对象,但不接受任何参数。常用于提供数据源。1
2Supplier<String> supplier = () -> "Hello";
String s = supplier.get();Runnable
:定义一个无参数无返回值的执行块。1
2Runnable runnable = () -> System.out.println("Hello World");
new Thread(runnable).start();
方法引用
方法引用是lambda
表达式的更简洁的语法,它允许我们引用类中的现有方法。
当使用方法引用时,目标引用放在分隔符::
之前,方法名称在它之后提供,即:目标引用::
方法。
使用 ::
运算符作为 Lambda
调用特定方法的缩写,并且拥有更好的可读性。
lambda 表达式与方法引用等价示例:
1 | (Apple a) -> a.getWeight() Apple::getWeight |
方法引用的三种类型
- 静态方法引用:如:
- lambda表达式:
(s) -> Integer.parseInt(s)
- 方法引用:
Integer::parseInt
- 任意类型的实例方法引用:引用一个类型的实例方法,并且这个实例作为参数传递给
lambda
表达式时使用。如:
- lambda表达式:
(s) -> s.toUpperCase()
- 方法引用:
String::toUpperCase
- 现有对象的实例方法引用:在
lambda
表达式中调用一个已经存在的对象的方法时使用。如:
- lambda表达式:
() -> expensiveTransaction.getValue()
- 方法引用:
expensiveTransaction::getValue
总结
Java 8
中的 Lambda
表达式可以让代码更加简洁、易读,并且提高开发效率。
Title: Java 8 In Action Lambda
Author: Amber
Date: 2024-07-01
Last Update: 2024-10-29
Blog Link: https://wyiyi.github.io/amber/2024/07/01/lambda/
Copyright Declaration: Copyright © 2022 Amber.