在代码中使用Lambda表达式
Logger logger = new Logger();
logger.debug(() -> "Look at this: " + expensiveOperation());
基本类型
- 流与数值流的转换
流转换为数值流
mapToInt(T -> int) : return IntStream
mapToDouble(T -> double) : return DoubleStream
mapToLong(T -> long) : return LongStream
IntStream intStream = list.stream().mapToInt(Person::getAge);
当然如果是下面这样便会出错
LongStream longStream = list.stream().mapToInt(Person::getAge);
- 数值流方法
下面这些方法作用不用多说,看名字就知道:
sum()
max()
min()
average() 等...
重载解析
Lambda 表达式作为参数时,其类型由它的目标类型推导得出,推导过程遵循如下规则:
- 如果只有一个可能的目标类型,由相应函数接口里的参数类型推导得出
- 如果有多个可能的目标类型,由最具体的类型推导得出;
- 如果有多个可能的目标类型且最具体的类型不明确,则需人为指定类型。
@FunctionalInterface
为了提高Stream 对象可操作性而引入的各种新接口,都需要有Lambda 表达式可以实现它。它们存在的意义在于将代码块作为数据打包起来。因此,它们都添加了@FunctionalInterface 注释。该注释会强制javac 检查一个接口是否符合函数接口的标准。如果该注释添加给一个枚举类型、类或另一个注释,或者接口包含不止一个抽象方法,javac 就会报错。重构代码时,使用它能很容易发现问题。
二进制接口的兼容性
Java 8 中为Collection 接口增加了stream 方法,这意味着所有实现了Collection 接口的类都必须增加这个新方法。对核心类库里的类来说,实现这个新方法(比如为ArrayList 增加新的stream 方法)就能就能使问题迎刃而解。缺憾在于,这个修改依然打破了二进制兼容性。
需要在Java 8 中添加新的语言特性:默认方法
默认方法
Collection 接口中增加了新的stream 方法,如何能让MyCustomList 类在不知道该方法的情况下通过编译? Java 8 通过如下方法解决该问题:Collection 接口告诉它所有的子类:“如果你没有实现stream 方法,就使用我的吧。”接口中这样的方法叫作默认方法,在任何接口中,无论函数接口还是非函数接口,都可以使用该方法。
public interface Parent {
public void message(String body);
public default void welcome() {
message("Parent: Hi!");
}
public String getLastMessage();
}
多重继承
三定律
如果对默认方法的工作原理,特别是在多重继承下的行为还没有把握,如下三条简单的定
律可以帮助大家。
- 类胜于接口。如果在继承链中有方法体或抽象的方法声明,那么就可以忽略接口中定义
的方法。 - 子类胜于父类。如果一个接口继承了另一个接口,且两个接口都定义了一个默认方法,
那么子类中定义的方法胜出。 - 没有规则三。如果上面两条规则不适用,子类要么需要实现该方法,要么将该方法声明
为抽象方法。
权衡
接口和抽象类之间还是存在明显的区别。接口允许多重继承,却没有成员变量;抽象类可以继承成员变量,却不能多重继承。
接口的静态方法
如果想创建一个由简单值组成的Stream,自然希望Stream 中能有一个这样的方法。这在以前很难达成,引入重接口的Stream 对象,最后促使Java 为接口加入了静态方法。
Stream 和其他几个子类还包含另外几个静态方法。特别是range 和iterate方法提供了产生Stream 的其他方式。
Optional
Optional 是为核心类库新设计的一个数据类型,用来替换null 值。人们对原有的null 值有很多抱怨,甚至连发明这一概念的Tony Hoare 也是如此,他曾说这是自己的一个“价值连城的错误”。
Optional 对象不仅可以用于新的Java 8 API,也可用于具体领域类中,和普通的类别无二致。当试图避免空值相关的缺陷,如未捕获的异常时,可以考虑一下是否可使用Optional对象。
Optional emptyOptional = Optional.empty();
Optional alsoEmpty = Optional.ofNullable(null);
assertFalse(emptyOptional.isPresent());
// 例4-22 中定义了变量a
assertTrue(a.isPresent());