写这篇文章的目的总结梳理一下读这本书的收获,未来用到java8某些冷门特性时进行回顾,
1、函数式编程思想
java8的核心改变是提出函数式编程的思想,书中把这称作jdk变革最大的一次,函数式编程的应用场景十分普遍,如过去我们筛选苹果,筛选绿的会写一个方法,筛选大的又要写一个方法,在函数式编程里我们只需要定义一个叫做筛选苹果的方法,把筛选条件作为参数传入;又比如我们更常见的getUserById和getUserByName这些,函数式编程只要getUser(Function<T,R> f)。这里我们说可以把方法看作一个参数传给另一个方法,这些函数参数本质是interface,jdk给出了一些常用的函数式接口,上一篇里有,我们自己定义函数式接口的话,只需要接口里有且只有一个非默认方法即可(java8新引入的接口方法默认实现,为了兼容之前的类库)。当然函数式编程的思想不仅仅如此,引用透明性,科里化,策略模式,模板方法,观察者模式,责任链模式,工厂模式中函数式思想的优势,这些需要未来有机会实践体会了。
2、直观的代码风格改变
java8对开发者带来最直观的改变就是lambda和stream,这两种东西改变了编程风格,这里先提前说一句lambda并不是匿名内部类的另一种表达方式,二者编译后的字节码都不一样,lambda创建额外的类被invokedynamic代替了(jdk7引入的)。stream更不是迭代器的另一种表达方式,stream可以表示无限流,延迟计算,有出色的并发能力。
2.1、Lambda
lambda是函数式接口的具体实现,形如(类型A 参数a,类型B 参数b) -> 表达式 或 (类型A 参数a,类型B 参数b) -> {表达式;return 结果},前边的小括号里要是只有一个参数可以省略类型声明和小括号,多个参数但类型一致的话也能省略类型声明,没参数的话只写小括号。“->”前边的代表函数传入的参数,后边的代表表达式返回的结果代表函数返回的结果,如Function<T,R>的Lambda签名就是就是 T -> R,Cosumer<T> 的就是T -> void,Supplier<T>就是 ()->T。
由于对并发性的支持,lambda内的变量要是final或近似final的,外部的引用变量可以出现再lambda内,只要引用不变就行,内容可以变,基础类型的话出现再lambda内不能修改,比如外面有个int i=0;lambda内部不能i++。
2.2、方法引用
方法引用也是一种函数实现,形如xxx::f,比如一个对象a中的一个方法是 public T f(R r);a::f的Lambda签名就是T -> R,这就可以让你自己写出方法来,然后通过方法引用当作函数形式的参数传入其他方法。
lambda表达式中若只有一个参数,表达式里只调用了该参数的方法,方法引用可直接协作参数类型的引用,如(Apple a) -> a.getWeight() 等效于 Apple::getWeight;
若只有一个参数(或没有),表达式里的方法也以这个参数为参数,方法引用可直接用表达式里的方法,如 (String s) -> System.out.println(s) 等效于System.out::println
其他方法引用的方式不一一列举了。
2.3、新的排序表达和比较器复合
过去的排序我们常用Collections.sort传入list和匿名内部类,java8在List类里加了sort方法,直接list.sort(