方法引用:

在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了一个对象的引用,然后你把这个引用传递到别的地方,你就可以调用这个对象里的属性和方法是一样的道理。

clipboard.png


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表达式的实际应用,好让大家更好的理解它们的用法。


Half
238 声望17 粉丝

The Long Way Round.