lambda表达式笔记

前几天一位好友分享了一篇文章,其中讲到了lambda表达式,正好最近看了一些内容,就做做笔记吧...

lambda表达式服务于函数式接口,如果需要一个函数式接口的对象时,就可以用lambda表达式代替

什么是函数式接口?

1.最主要的一点是只包含一个抽象方法的接口(Object类的方法不算啊)
2.带有@FunctionalInterface注解的一定是函数式接口,但是不代表没有该注解就表示不是,只要是一个接口,有且只有一个抽象方法,那么就是函数式接口,注解配合编译器可以很好的辅助,我们在自定义函数式接口的时候,也会更加直观,有助于代码的可读性

几个简单的例子

Arrays.sort方法需要一个实现了Comparator接口的实例,正常情况下,我们需要创建一个实例,然后实现Comparator接口,然后将该实例传入Arrays.sort方法中,还是比较繁琐的,但是使用lambda表达式就可以非常简化

编译器如果能推导出参数类型,小括号中的参数类型是可以省略的

        String[] array = new String[3];
        array[0] = "ccc";
        array[1] = "bb";
        array[2] = "a";
        System.out.println(array[0]);// 输出ccc
        Arrays.sort(array, (first, second) -> first.length() - second.length());
        System.out.println(array[0]);// 输出a

对于逻辑比较复杂的代码,逻辑处理部分可以使用{}包起来

        Timer t = new Timer(1000, event -> {
            System.out.println("action");
            System.out.println("listener");
        });
        t.start();

表达式也是可以当作一个参数传递的

        String[] array = new String[3];
        array[0] = "ccc";
        array[1] = "bb";
        array[2] = "a";
        System.out.println(array[0]);
        Comparator<String> comp = (first, second) -> first.length() - second.length();
        Arrays.sort(array, comp);
        System.out.println(array[0]);

如果没有入参的时候,小括号不能省略

        Thread thread = new Thread(() -> {
            for (int i = 0; i < 100; i++) {
                System.out.println(i);
            }
        });
        thread.start();

现在工作主要是用Scala语言,Scala在语法等方面对比Java来讲更加函数化,遍历、排序、过滤等操作甚至来讲只用一行代码就可以完成

但是lambda作为Java8的一个大更新,所以还是有必要学习一下新特性的

 

下面新增一些关于Supplier和Consumer的知识

Supplier也是一个函数式接口,这个接口只有一个返回泛型T的方法

 

 因此如果我们想要得到某个实例,就可以使用这个函数式接口了比如

        // 这个实体有两个构造,一个有参,一个无参
Supplier<LengthComparator> lcp1 = () -> new LengthComparator("a", "b");
// 直接使用构造方法引用,编译器会自动帮我们引用到无参上面去,但是如果我们把无参给注释掉,这里就会报错了 Supplier
<LengthComparator> lcp2 = LengthComparator::new; System.out.println(lcp1.get()); // 打印这个对象的地址 Supplier<String> str = () -> "hello"; System.out.println(str.get()); // 打印hello

 

Consumer也是一个函数式接口,其中有两个方法:

  一个方法是传个泛型参数,没有返回值;

  还有一个是链式方法;

 

        /**
         * void accept(T t);
         */
        Consumer<String> sss = (x) -> System.out.println("x = " + x);
        sss.accept("hello"); // x = hello

        /**
         * default Consumer<T> andThen(Consumer<? super T> after)
         */
        Consumer<String> ssss = (x) -> System.out.println(x);
        ssss.andThen(sss); // 没有任何输出
        ssss.andThen(sss).accept("world"); // 先执行ssss的accept方法,再执行sss的accept方法;

由上我们可以看出,Consumer这个接口对于批量操作数据非常的方便,比如有一批数据,需要经过多种逻辑运算,就可以链式使用,这个非常类似于Scala的函数式编程风格,比如list.map(x => x.name).filter(name => name.equals("xxx"))

需要注意的是,链式玩法中,Consumer的泛型必须都是统一类型,因为accept参数需要被多次用到,如果类型不一致,编译报错;

posted @ 2019-09-08 14:04  dadapeng  阅读(486)  评论(0编辑  收藏  举报