方法引用:
在Java 8之前只能进行值传递,方法是不能传递的。如果你想调用一个方法你必须先获取到它所在的类的实例,然后再通过实例去调用这个方法,但是Java 8新增了方法引用这个新特性可以让你直接把方法当做值来传递。
1.下面这段代码代码的作用是遍历获取目录下所有的文件和目录,并且还加了一个筛选条件,只筛选出不隐藏的文件和目录,这里我们其实只是想调用FileFilte中的accept方法来进行筛选,但是我们需要先创建FileFilter的匿名对象,然后重写整个accept方法,这样我们才调用到了这个方法,其中只有第三行代码是会有变化的,其他的代码都是固定的,但是我们每次还是要把其他固定的模板代码重新写一遍。
File[] hiddenFiles = new File("F:\\test").listFiles(new FileFilter() {
public boolean accept(File file) {
return !file.isHidden();
}
});
2.现在Java 8中的方法引用就解决了这个问题,让我们看下列的代码,我们发现匿名类和重写方法的步骤都已经没有了,上述代码的本质其实就是调用传进来的File对象的isHidden方法,现在File:: isHiden
这个写法就是和上面的代码是同样的作用,但是代码精简了很多,那些无用的冗余代码都不见了。
File[] hiddenFiles = new File("F:\\test").listFiles(File::isHidden);
3.我们从源码来看看listFiles
方法做了什么操作,而这两种写法又有什么不同。
-
首先listFiles方法接受了一个FileFilter类型的对象,
list
方法是获取所有的文件,files是用来存储筛选之后的元素,循环所有获得到的文件数组,然后调用FileFilter中的accept方法来进行条件筛选,放入files后返回。public File[] listFiles(FileFilter filter) { String ss[] = list(); if (ss == null) return null; ArrayList<File> files = new ArrayList<>(); for (String s : ss) { File f = new File(s, this); if ((filter == null) || filter.accept(f)) files.add(f); } return files.toArray(new File[files.size()]); }
-
再看看FileFilter对象是什么,发现它是一个接口,所以Java 8之前的写法都是写了个匿名对象来实现这个接口,重写它的accept方法。看到这里其实很明显了,这就是一个策略模式的应用。而方法引用就是让我们直接把需要在
accept
方法里调用的方法传递进去,不需要像以前一样来个全家桶写一堆固定模板。@FunctionalInterface public interface FileFilter { boolean accept(File pathname); }
4.下面的图介绍了Java 8之前和之后这段代码的逻辑流程,在Java 8之前是需要先创建FileFilter匿名对象然后再调用File.listFiles
方法,而现在只需要File::isHiden
写法就可以达到同样的目的,其实它的含义就是创建了一个方法引用,所以你可以通过传递引用来传递这个方法,就好像你new了一个对象的引用,然后你把这个引用传递到别的地方,你就可以调用这个对象里的属性和方法是一样的道理。
Lambda-匿名函数:
上面的方法引用让我们可以把方法也当做值来进行传递,但是有时候我们传递进去的代码并没有像File.isHidden
方法一样封装起来,而这种情况也是经常发生的,有时候为了一个特殊需求我需要写段代码来解决,但是次数用的极少,没必要封装个方法,而Lambda表达式则解决了这个问题。
1.比如我们想要筛选出一个文件名叫abc.txt的文件,我们可以这样写,我们看到 file.getName(). equals("abc.txt"));
是我们自己写出来的,我们并没有把它封装成方法就拿来使用了,注意看->
符号前面,那个代表的是上面accept
方法接受的参数,而->
后面则是我们拿传递进来的参数来操作,只是要确保你的这行代码返回的类型是要和accept
方法的返回类型一致。
File[] files3 = new File("F:\\test").listFiles((File file) -> file.getName().equals("abc.txt"));
2.以上只是Lambda表达式最简单的方式,我们还可以有下列这些更加多样化的操作,Lambda表达式在Java 8中是非常重要的部分,因为后面我们讲到的stream(流)就是基于它来使用的。
//单个参数的时候可以省略括号和类型,直接写形参
File[] files4= new File("F:\\test").listFiles((file -> file.getName().equals("abc.txt")));
//如果要写多行代码,可以加大括号把所有代码括起来,最后你同样需要返回正确的类型
File[] files5= new File("F:\\test").listFiles(
(File file) ->
{
boolean flag = file.getName().equals("d") && file.getName().contains("d");
return flag;
}
);
下面我会增加一篇文章关于Lambda表达式的实际应用,好让大家更好的理解它们的用法。
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。