一. 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);