教你看懂System.out::println

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

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

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

第一印象, 哇, 好高大上的写法, 那么这究竟是怎样的一种语法呢。
我们一步一步来探究:
首先, 我们看一下是java.lang.Iterable下的一个默认方法forEach调用的, 好家伙, 一看到这个function包下面的被@FunctionalInterface注解声明的Consumer接口, 瞬间就了然了, 这不又是函数式编程搞的鬼么?(如果大家不理解什么是函数式编程, 可以去看看我的一篇博客 —— 深入浅出讲解Optional包装类, 里面有详细的介绍和代码 )。
现在的问题应该很明朗了, System.out::println这段代码其实就是Consumer接口的一个实现方式啊。 具体是怎么实现的, 我们再码一段代码。

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

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

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

/**
 * 打印加强处理类
 * 

* * @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 strList = Arrays.asList("YangHang", "AnXiaoHei", "LiuPengFei");

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

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

我叫杨小邪, 大家有任何问题都可以给我留言。

推荐文章:
Java程序员必知的高级IDEA调试技巧

你可能感兴趣的:(JAVA基础)