现在看一个如何从传统的Java程序到Java8风格的程序的替换的例子。
需求: 想要对仓库的苹果按照重量进行排序。
思路:当前Java8的List类已经提供了一个sort方法,其签名如下:
void sort(Comparator<? super E> c)
从签名中可以看到其入参为一个Comparator类型的对象,也就是说需要传递一个排序策略即可实现该排序,排序策略的不同,其排序的行为即不同。
方法一、实现Comparator
public class AppleComparator implements Comparator<Apple> {
public int compare(Apple a1, Apple a2){
return a1.getWeight().compareTo(a2.getWeight());
}
}
inventory.sort(new AppleComparator());
该方法中我们定义了一个类AppleComparator来实现Comparator接口,然后用这个类定义一个对象传给sort。
方法二、使用匿名类
方法一中定义了一个类确只实例化一次,我们可以通过使用匿名类的方式来解决。
inventory.sort(new Comparator<Apple>() {
public int compare(Apple a1, Apple a2){
return a1.getWeight().compareTo(a2.getWeight());
}
});
前面两种方法都是我们在Java8之前习惯用到的两种方式,使用匿名类的方式虽然相对简单些,但还是存在了一些冗余代码。
方法三、使用Lambda表达式
Java8引入了Lambda表达式,它可以方便地传入代码,只要入参为函数式接口。而本例中Comparator就是一个函数式接口,代表了函数描述符(T, T) -> int。本例中泛型T指的是苹果,即(Apple, Apple) -> int。现在使用了Lambda表达式的代码如下:
inventory.sort((Apple a1, Apple a2) -> a1.getWeight().compareTo(a2.getWeight())
);
另外,我们知道Java编译器可以根据Lambda上下文推断出参数类型,所以我们可以把上面的式子简化为:
inventory.sort((a1, a2) -> a1.getWeight().compareTo(a2.getWeight()));
代码还能再简单点吗?由于Comparator具有一个叫作comparing的静态辅助方法,它可以接受一个Function来提取Comparable键值,并生成一个Comparator对象。所以上面的代码还可以再简化为:
inventory.sort(comparing((a) -> a.getWeight()));
现在看上去代码相对方法一和方法二而言,已经简单很多了。
方法四、使用方法引用
通过学习,我们知道方法引用可以代替Lambda式的参数,使得代码更简单,此处我们也可以把方法三的代码再进一步简化:
inventory.sort(comparing(Apple::getWeight));
到此为止,我们已经用了Java8的语法来实现了对仓库里的苹果按重量进行排序的需求。经过一步一步地演化,我们可以看出Java8的代码相对以前的代码来说,其优点是:
1、代码简短;
2、意思直观明显;
3、代码阅读起来贴近需求描述;