一. Lambda定义(λ):
-- 匿名,它不像普通方法那样有一个明确的名称;
-- 函数,它不像普通方法那样属于某个特定的类,但和方法一样,Lambda有参数列表、函数主体、返回类型或抛出异常列表:
-- 传递,Lambda可以作为参数传递给方法或存储在变量中:
-- 简洁。
二. Lambda表达式结构:
1. 参数列表;
2. 箭头:箭头->把参数列表与Lambda主体分隔开;
3. Lambda主体:表达式就是Lambda表达式的例子。
三.Lambda基本语法:
(parameters) -> expression
或
(parameters) -> { statements; }
使用显式返回语句时需要使用花括号“{}”。
eg: Lambda示例:使用案例Lambda示例
无参数,返回void() -> {}
无参数,返回String() -> "Raoul"
无参数,返回String(利用显式返回语句)() -> { return "Result";}
布尔表达式(List list) -> list.isEmpty()
创建对象() -> new Apple(10);
消费一个对象(Apple apple) -> { System.out.println(a.getWeight()); }
从一个对象中选择/抽取(String s) -> s.length()
组合两个值(int a, int b) -> a * b
比较两个对象(Apple a1, Apple a2) -> a1.getWeight().compareTo(a2.getWeight())
四. 在哪里使用Lambda
可以在函数式接口上使用Lambda表达式。
函数式接口:只定义一个抽象方法的接口。
@FunctionalInterface:表明为函数式接口,但它不是必须的。
(T, U) -> R表达式展示了应当如何思考函数描述符。左侧代表了参数类型。这里它代表一个函数,具有两个参数,分别为泛型T和U,返回类型为R。
Java的泛型只能绑定到引用类型,当需要引用原始类型时因为自动装箱和拆箱机制时,装箱后的值需要更多的内存,需要付出性能代价,为了避免装箱操作对Predicate和Function等通用函数式接口的原始类型特化:IntPredicate、IntToLongFunction等。
Java8中的常用函数式接口:函数式接口函数描述符原始类型特化
PredicateT -> booleanIntPredicate,
LongPredicate,
DoublePredicate
ConsumerT -> voidIntConsumer,
LongConsumer,
DoubleConsumer
FunctionT -> RIntFunction,
IntToDoubleFunction,
IntToLongFunction,
LongFunction,
LongToDoubleFunction,
LongToIntFunction,
DoubleFunction,
ToIntFunction,
ToDoubleFunction,
ToLongFunction
Supplier() -> TBooleanSupplier,
IntSupplier,
LongSupplier,
DoubleSupplier
UnaryOperatorT -> TIntUnaryOperator,
LongUnaryOperator,
DoubleUnaryOperator
BinaryOperator(T, T) -> TIntBinaryOperator,
LongBinaryOperator,
DoubleBinaryOperator
BiPredicate(L, R) -> boolean
BiConsumer(T, U) -> voidObjectIntConsumer,
ObjectLongConsumer,
ObjectDoubleConsumer
BiFunction(T, U) -> RToIntBiFunction,
ToLongBiFunction,
ToDoubleBiFunction
Runnable() -> void
当需要Lambda表达式抛出异常时,有两种方式:
-- 自己编写新的函数式接口,并声明受检异常(任何函数式接口都不允许抛出受检异常);
-- 将Lambda包在一个try/catch块中。@FunctionalInterface
public interface BufferedReaderProcessor {
String process(BufferedReader b) throws IOException;
}Function f = (BufferedReader b) -> {
try {
return b.readLine();
} catch (IOException e) {
throw new RuntimeException(e);
}
};
当Lambda表达式抛出一个异常时,throws语句也必须与Lambda所指类型相匹配。
如果Lambda的主体是一个语句表达式,它就和一个返回void的函数描述符兼容(当然需要参数列表也兼容)。
eg:尽管List的add方法返回的是boolean,但以下两行都是合法的://Predicate返回了一个boolean
Predicate p = s -> list.add(s);
//Consumer返回了一个void
Consumer b = s -> list.add(s);
Lambda类型推断:Java编译器会从上下文(目标类型)中推断出用什么函数式接口来配合Lambda表达式,所以也能推断出适合Lambda的签名,因为函数描述符可以通过目标类型来得到。这样就可以在Lambda中省去标注参数类型,当参数只有一个时还可以省去参数的括号。
Lambda使用局部变量的限制:Lambda表达式对值封闭,而不是对变量封闭。Lambda表达式引用的局部变量必须是final的,只能有一次赋值。
这是因为实例变量是储存在堆中,而局部变量是储存在栈上。如果Lambda可以直接访问局部变量,而且Lambda是在一个线程中使用的,则使用Lambda的线程可能会在分配该变量的线程将这个变量收回之后,去访问该变量。因此,Java在访问自由局部变量时,实际上是在访问他的副本而不是访问原始变量。
Lambda方法引用:
方法引用就是Lambda的快捷写法。目标引用放在分隔符::前面,方法的名称放在后面。
eg:Lambda表达式等效的方法引用
(Apple a) -> a.getWeight()Apple::getWeight
() -> Thread.currentThread().dumpStack()Thread.currentThread()::dumpStack
(str, i) -> str.substring(i)String::substring
(String s) -> System.out.println(s)System.out::println
如何构建方法引用:
主要有三类:
-- 指向静态方法的放方法引用;
-- 指向任意类型实例方法的方法引用;
-- 指向现有对象的实例方法的方法引用。
Lambda构造函数引用:
对于一个现有构造函数,可以利用它的名称和关键字new来创建它的一个引用:ClassName::new。它的功能与静态方法的引用类似。Lambda表达式等价构造方法引用Lambda表达式
Supplier c1 = Apple::new;
Apple a1 = c1.get();Supplier c1 = () -> new Apple();
Apple a1 = c1.get();
Function c2 = Apple::new;
Apple a2 = c2.apply(110);Function c2 = (weight) -> new Apple(weight);
Apple a2 = c2.apply(110);
BiFunction c2 = Apple::new;
Apple a2 = c2.apply("green", 110);BiFunction c2 = (color, weight) -> new Apple(color, weight);
Apple a2 = c2.apply("green", 110);
复合Lambda表达式的有用方法:复合类型方法说明举例
比较器复合reversed()逆序//按重量递减排序
inventory.sort(comparing(Apple::getWeight))
.reversed();
thenComparing比较器链(接受一个函数作为参数,如果两个对象用第一个Comparator比较之后是一样的,就提供第二个Comparator//两个苹果一样重时按国家排序
inventory.sort(comparing(Apple::getWeight))
.reversed().
thenComparing(Apple::getCountry);
谓词复合
(and和or方法的优先级是按照在表达式链中的位置,从左向右确定的)negate非//产生现有对象redApple的非
Predicate notRedApple = redApple.negate();
and将两个Lambda用and组合起来//一个苹果既是红色又比较重(链接两个谓词来生成一个Predicate对象)
Predicate RedAndHeavyApple = redApple.and(a -> a.getWeight() > 150);
or要么//要么重(150g以上)的红苹果,要么是绿苹果
Predicate RedAndHeavyApple = redApple.and(a -> a.getWeight() > 150)
.or(a -> "green".equals(a.getColor()));
函数复合andThen先对输入应用一个给定函数,再对输出应用另一个函数//等同于数学上的g(f(x)),返回4
Function f = x -> x + 1;
Function g = x -> x * 2;
Function h = f.andThen(g);
int result = h.apply(1);
compose先把给定的函数用作compose的参数里面给的那个函数,然后再把函数本身用于结果//等同于数学上的f(g(x)),返回3
Function f = x -> x + 1;
Function g = x -> x * 2;
Function h = f.compose(g);
int result = h.apply(1);