从java7到java8,最主要的变化可以总结为
□Lambda表达式
□ 方法引用
□流和默认方法
让我们通过一个小例子感受一下
情景 1 集合对象排序 (对list中的苹果按照重量排序)
- Collections.sort(list,new Comparator<Apple>(){
- public
int compare(Apple a1,Apple a2){ - return a1.getWeight()-a2.getWeight();
- }
- });
在java8里面,这种实现只需要一行很简短的代码
- list.sort(java.util.Comparator.comparing(Apple::getWeight))
这其中涉及到lambda表达式,接口中静态方法,方法引用等java8新特性,而且一看就明白其功能,给list排序,排序依据weight
Java多线程历程 java1.0 线程和锁------à java5 线程池和并发集合----------à java7 添加分支/合并(fork/join)--à java8 添加stream支持并发处理数据
在java8中加入Streams可以把 接口默认方法和把代码块传递给方法的简介方式(方法引用、lambda表达式)
把代码传递给方法 即是一种行为参数化思想,和函数式编程思想极为相近。
情景2 对目录文件过滤,筛选所有隐藏文件
- File [] hiddenFiles=new File(".").listFiles(new FileFilter(){
- public boolean accept(File file){
- return file.isHidden();
- }
- });
使用方法引用,直接将方法当做参数传递进去
- File [] hiddenFiles=new File(".").listFiles(File::isHidden)
既然对象File已经有了一个isHidden方法,我们不必再把它包含在啰嗦的匿名内部类里面过滤了。
与用对象引用(对象引用是用new创建的)类似,写下File::isHidden就创建一个方法引用,可以传递它
如可以写成这样
- FileFilter filter=File::isHidden;
- File [] isHiddenFiles2=new File(".").listFiles(filter);
从第一行代码可以看出,方法引用确实可以当做一个变量来使用
情景3 lambda传递条件代码 (筛选符合条件的苹果)
- public static List<Apple> filterGreenApple(List<Apple> list){
- List<Apple> result=new ArrayLst<>();
- for(Apple apple : list){
- if("green".equals(apple.getColor())){
- result.add(apple); //筛选颜色为绿色的苹果
- }
- }
- return result;
- }
- public static List<Apple> filterWeighterApple(List<Apple> list){
- List<Apple> result=new ArrayLst<>();
- for(Apple apple : list){
- if(apple.getWeight()>150){
- result.add(apple); //筛选重量大于150g的苹果
- }
- }
- return result;
- }
如果客户要筛选红苹果,筛选红且重量大于150g的苹果,…难道每次都要添加一个函数吗,而且这些函数除了条件筛选不一样,其他部分都是一样的。
首先我们可以把公用代码提出来
- public static List<Apple> filterApple(List<Apple> list){
- List<Apple> result=new ArrayList<>();
- for(Apple apple:list){
- if(....){ //苹果筛选条件
- result.add(apple);
- }
- }
- return result;
- }
- 进一步,我们可以把筛选条件传递进去
过滤函数可以改成这样 - public static List<Apple> filterApple(List<Apple> list,Predicate<Apple> p){
- List<Apple> result=new ArrayList<>();
- for(Apple apple:list){
- if(p.test(apple)){
- result.add(apple);
- }
- }
- return result;
- }
- Public interface Predicate<Apple>{
- Boolean test(Apple apple);
- }
即过滤函数有两个参数,一个时过滤的对象集合,一个时过滤的 条件对象(Predicate),而该条件对象应该有个可以返回Boolean的函数来确定是否符合条件(策略设计模式)
这样设计之后,把filterApple方法迭代集合的逻辑和要应用到集合中每个元素对象上的行为(谓词)区分开了。
比如要筛选出绿苹果
- public class GreenColorPredicate<Apple> implements Predicate<Apple>{
- public boolean test(Apple apple){
- if("green".equals(apple.getColor())){
- return true;
- }
- return false;
- }
- }
- GreenColorPredicate<Apple> greenPredicate=new GreenColorPredicate<>();
- filterApple(list,greenPredicate);
或者lambda表达式
- filterApple(list,(Apple apple)->"green".equals(apple.getColor()));
我们可以将上面的函数抽象化成泛型函数,这样就可以过滤香蕉,橘子等对象集合啦!!
行为参数化最大好处就是可以适应不断变化的需求。
总结:
1 行为参数化,就是一个方法接受多个不同的行为作为参数,在内部使用它们,完成不同行为的能力。
2 行为参数化可以让代码更好适应不同需求,减轻未来工作量。
3 传递代码,即将新行为作为参数传递给方法。可以用lambda表达式或者方法引用。
4 java api包含很多可以用不同行为进行参数化的方法,最常用就是排序,线程等。