美文网首页java8的世界Java8
Java 8之lambda表达式(二)

Java 8之lambda表达式(二)

作者: Kotyo | 来源:发表于2017-07-21 10:29 被阅读35次

    前言

    上一章咱们了解了以下内容:
    ① 为什么使用lambda表达式
    ② lambda表达式的语法
    ③ 函数式接口

    这一章咱们继续了解lambda表达式。

    1.4 方法引用

    咱们先看一下方法引用和非方法引用的区别:

    //非方法引用
    button.setOnAction(event->Sysout.out.println(event));
    //方法引用
    button.setOnAction(Sysout.out:println);
    

    正如上面所示:
    “ :: ”操作符将方法名和对象或类分隔开来。以下是三种主要情况:
    ① 对象::实例方法
    ② 类::静态方法
    ③ 类::实例方法
    在①和②中方法引用等同于方法参数的lambda表达式。如之前所述,Sysout.out::println等同于Sysout.out.println(x)。相似地,Math::pow等同于(x,y)->Math.pow(x,y)。
    在③中,第一个参数会成为执行方法的对象。例如:String::compareToIgnoreCase等同于(x,y)->x.compareIgnoreCase(y)。

    注意:如果有多个同名的重载方法,那么编译器会从上下文中找到最匹配的一个方法。例如,如果有两个版本的Math.max方法,一个接收整型作为参数,而另一个接收Double类型作为参数,究竟会选择哪个方法,取决于Math.max被转换为函数式接口的方法参数。同lambda表达式类似,方法引用也不会独立存在,它们经常被用于转换为函数式接口的实例。

    你还可以捕获方法引用中的this参数。例如,this::equals等同于x->this.equals(x)。当然你也可以使用super对象,例如,super::实例方法
    以下用来说明该机制的示例:

    class Greeter{
      public void greet(){
          Sysout.out.println("Hello World");
      }
    }
    
    class ChildGreeter extends Greeter{
      public void greet(){
        Thread t=new Thread(super::greet);
        t.start();
      }
    }
    

    当线程启动时,会调用它的Runnable方法,然后执行super::greet并调用父类中的greet方法。

    注意:在匿名类中,你可以调用一个闭合类或父类的员工方法,例如闭合类.this::方法或闭合类.super::方法。

    1.5 构造器引用

    构造器引用同方法引用类似,不同的是构造器引用中方法名是new。例如:Button:new表示Button类的构造器引用。
    你可以使用数组类型来编写构造器引用。例如:int[]::new是一个含有一个参数的构造器引用,这个参数就是数组的长度。它等同于lambda表达式x->new int[x]。数组构造器可以用来绕过java中的一个限制。在Java中,无法构造一个泛型类型T的数组。表达式new T[n]是错误的,因为它会被擦除为new Object[n]。这对于编写API的开发人员来说是一个问题。例如,我们希望构造一组按钮,Stream接口中有一个返回Object数组的toArray方法:

    Object[] buttons=stream.toArray();
    

    但是这并不能让我们满意。用户希望一组按钮对象而并不是一组Object对象,Stream API通过构造器引用解决了这个问题。它允许将Button[]::new 传给toArray方法:

    Button[] buttons=stream.toArray(Button[]::new);
    

    toArray方法会调用该构造器来获得一个正确类型的数组。然后它会填充并返回该数组。

    相关文章

      网友评论

        本文标题:Java 8之lambda表达式(二)

        本文链接:https://www.haomeiwen.com/subject/tkeskxtx.html