在不经意间, 我们会看到这样的代码

// 创建出一个数组
List<String> strList = Arrays.asList("YangHang", "AnXiaoHei", "LiuPengFei");

strList.forEach(System.out::println);

第一印象, 哇, 好高大上的写法, 那么这究竟是怎样的一种语法呢。

我们一步一步来探究:

首先, 我们看一下是​​java.lang.Iterable<T>​​下的一个默认方法​​forEach​​调用的, 好家伙, 一看到这个function包下面的被​​@FunctionalInterface​​注解声明的​​Consumer<T>​​接口, 瞬间就了然了, 这不又是函数式编程搞的鬼么?(如果大家不理解什么是函数式编程, 可以去看看我的一篇博客 —— 深入浅出讲解Optional包装类, 里面有详细的介绍和代码 )。

现在的问题应该很明朗了, ​​System.out::println​​这段代码其实就是​​Consumer<T>​​接口的一个实现方式啊。 具体是怎么实现的, 我们再码一段代码。

@Test
public void testDemo2() {
List<String> strList = Arrays.asList("YangHang", "AnXiaoHei", "LiuPengFei");

strList.forEach(x -> {
System.out.println(x);
});
}

然后, 我们惊喜的发现和上面的代码运行结果是一制的, 我们基本上可以断定, 上面那种写法是下面这种的一种缩写形式。 就是把你遍历出来的每一个对象都用来去调用System.out(也就是PrintStream类的一个实例)的println方法。

最后, 大家是不是有一个想法, 想自己写一个​​Consumer<T>​​接口的实现类, 让foreach调用一下。

/**
* 打印加强处理类
* <p>
*
* @author YangHang
* @Date 2018年12月9日 下午12:25:30
*
*/
public class PrintUtil {

/**
* 对要遍历的元素添加add操作
*/
public void addString(String x) {
System.out.println(x + "add");
}
}

然后, 我们这么来玩

@Test
public void testDemo3() {
List<String> strList = Arrays.asList("YangHang", "AnXiaoHei", "LiuPengFei");

strList.forEach(new PrintUtil()::addString);
}

运行一下, 果然可以。

但是我发现, 如果是静态方法的时候必须得用类名双冒号静态方法, 这估计是语法的一种, 发夹注意就好。


我的个人公众号,关注之后可以向我留言,比如关于我进阿里的面试知识,有简历、学习路线的问题也欢迎询问我

教你看懂System.out::println_函数式编程