关于 JDK8 中方法引用的一个问题

作者:RednaxelaFX
链接:https://www.zhihu.com/question/45218076/answer/98632631
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

方法引用是当你想把一个方法当作一个“函数指针”传给别的方法用时有用的。

例如说,我有个ArrayList想把里面每个元素都打印出来,每个元素一行。
那么Java 8之前会这样写:
  for (ElementType e : list) {
    System.out.println(e);
  }
从Java 8开始,使用ArrayList的新API加上lambda表达式,我们可以这样写:
  list.forEach(e -> System.out.println(e));
而这里的lambda表达式的内容其实只不过就是把参数传给了println()方法,而没有做任何别的事情,所以可以进一步简写为:
  list.forEach(System.out::println);

仅此而已。

重点:
  • System.out是一个PrintStream实例的引用;System.out::println 是对一个实例方法的引用
    • 该引用同时指定了对实例(System.out)的引用以及对方法(PrintStream::println)的引用
  • System.out::println 不是 System.out.println 的等价物;前者是一个方法引用表达式,而后者不能单独作为一个表达式,而必须在后面跟上由圆括号包围的参数列表来构成方法调用表达式。
  • System.out::println 可以看作 lambda表达式 e -> System.out.println(e) 的缩写形式。
作者:陆萌萌
链接:https://www.zhihu.com/question/45218076/answer/125250908
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

无意间看到了,还是答一下吧,这个时间点上我想题主如果继续学习的话应该早已经明白了问题的答案,所以这篇答案就算留给后面无意间看到而又对此困惑的朋友吧:

实际上这个方法引用仅仅是一个Lambda表达式简化写法的语法糖,System.out::print和System.out.print()的语义完全不同,就像R大所说,用::而不用.就是为了区分语义,即使用了“::”都造成了困惑,那么用“.”的话恐怕困惑会更严重吧。

如果后来的朋友看不明白这个解释的话,那么首先就要明白Java8增加的这个Lambda表达式是干什么的,其实Lambda表达式主要解决的问题就是“如何把一个动作传入一个方法?”在C/C++中,我们可以给函数传入一个函数指针来做到这一点,C#中我们可以将一个委托传入方法中;本质上即是将一个方法/函数当成参数传给另一个方法/函数,那么之前Java是没有类似的语法的,一直以来Java中是如何解决这个问题的呢?

事实上Java一直以来的解决方案都是在方法参数上接收一个接口,接口中有需要的动作(即方法),调用的时候只要传入一个实现这个接口的对象即可间接地把方法传进去,例如创建线程的代码:
new Thread(new Runnable() {
                @Override
                public void run() {
                    //动作             
                }
            }).start();
上面这个代码,我们绕了这么大一圈,只是为了把run()传进Thread的构造器而已,我们能不能直接把run()传入呢?这里就出现了Java 8的Lambda表达式,实际上Lambda表达式就是一个匿名方法,它是这种形态的:
(方法参数列表) -> {方法体}
所以上面Thread的例子就能写成:
new Thread(() -> {
                //动作                
            }).start();
因为run()的参数是空的,所以前面的括号不能省略,箭头后面的方法体如果的单行的,可以省略{}大括号。回到这道题,如果出现了传入的匿名方法有一个参数,而这个方法的方法体中又调用了另一个方法,刚好把当前的参数传入这个方法中,例如:
List<String> list = Arrays.asList("真","可","爱");
list.forEach((String item) -> System.out.print(item));
//这里的参数列表具有一定的类型推倒功能,所以也可以进一步省略写为:
list.forEach(item -> System.out.print(item));
Java 8中List多了forEach方法,用于遍历列表,而遍历时做的事情会被当作动作传入这个方法,如果您只是打印里面的值的话,就会写成上面的样子,而用到本题的语法糖中就是:
List<String> list = Arrays.asList("真","可","爱");
list.forEach(System.out::print);

既然只是想把每次遍历得到的String item传入print方法,那么简略的用一个“方法引用”来引用这个方法就行了,这里这个System.out::print仅仅是等效于item -> System.out.print(item),和System.out.print()的语义是不同的,这里仅仅表达的是将唯一的参数传入System.out这个流对象的print方法的Lambda表达式而已,所以这里也不加括号;System.out.print()的语义就是直接调用方法了。这个语法糖除了用于普通方法外,如果是构造函数的方法引用,那么就是“ClassName::new”这种写法。这就是上面这个现象我自己的理解。

如果您对此还想继续深入学习的话,建议您学习Java 8增加的Stream<T> API,它是Java对于函数式编程的一个实践,学习它有助于理解函数式编程的一些基本思想,并对Scala、Spark编程的学习有一定好处。