lambda表达式3 – 方法引用

一个实例

使用lambda表达式可以创建简洁的匿名方法。不过,有时候lambda表达式只是简单的调用了已有的方法。此时,使用方法引用无疑是一个更简洁易读的方案。

再来看看之前使用过的Person类:

public class Person {

    public enum Sex {
        MALE, FEMALE
    }

    String name;
    LocalDate birthday;
    Sex gender;
    int age;
    String emailAddress;

    public int getAge() {
        return age;
    }

    public LocalDate getBirthday() {
        return birthday;
    }

    public static int compareByAge(Person a, Person b) {
        return a.birthday.compareTo(b.birthday);
    }
}

假设所有的Person对象都保存在一个数组中,然后想按年龄对数组成员进行排序,可以使用如下的代码进行实现:

        Person[] rosterAsArray = roster.toArray(new Person[roster.size()]);

        class PersonAgeComparator implements Comparator {
            public int compare(Person a, Person b) {
                return a.getBirthday().compareTo(b.getBirthday());
            }
        }

        Arrays.sort(rosterAsArray, new PersonAgeComparator());

这里调用的sort方法的声明是这样子的:

static  void sort(T[] a, Comparator c)

注意这里的Comparator是一个函数式接口,因此无需再定义一个PersonAgeComparator类并创建一个实例,直接使用lambda表达式实现即可:

        Arrays.sort(rosterAsArray,
                (Person a, Person b) -> {
                    return a.getBirthday().compareTo(b.getBirthday());
                }
        );

不过Person类中已经有了一个compareByAge方法,因此可以对上面的表达式作进一步的简化:

        Arrays.sort(rosterAsArray,
                (a, b) -> Person.compareByAge(a, b)
        );

因为这个lambda表达式只是调用了一个已有的方法,因此可以使用方法引用替换lambda表达式:

        Arrays.sort(rosterAsArray,  Person::compareByAge);

这里的方法引用Person::compareByAge和lambda表达式(a, b) -> Person.compareByAge(a, b)在语义上是一样的。它们都有如下的特性:

  • 参数列表copy自Comparator<Person>.compare方法,即(Person, Person);
  • 调用了Person.compareByAge方法。

在这个例子以及下面的示例中可以看到,方法引用使用的场合大致是和lambda表达式重合的。

方法引用的类型

方法引用有四种类型:

类型

示例

静态方法引用

ContainingClass::staticMethodName

实例方法引用

containingObject::instanceMethodName

一个类任意对象的实例方法引用

ContainingType::methodName

构造方法引用

ClassName::new

静态方法引用

前面示例中的Person::compareByAge就是一个静态方法引用。

实例方法引用

下面的代码演示了实例方法引用:

class ComparisonProvider {
    public int compareByName(Person a, Person b) {
        return a.getName().compareTo(b.getName());
    }

    public int compareByAge(Person a, Person b) {
        return a.getBirthday().compareTo(b.getBirthday());
    }
}

ComparisonProvider myComparisonProvider = new ComparisonProvider();
Arrays.sort(rosterAsArray, myComparisonProvider::compareByName);

方法引用myComparisonProvider::compareByName中调用的方法compareByName是对象myComparisonProvider的一部分。JRE会推断出方法的参数类型。在这个例子里就是(Person, Person)。

一个类的任意对象的方法引用

如下的代码演示了一个类的任意对象的实例方法引用:

        String[] stringArray = {"Barbara", "James", "Mary", "John",
                "Patricia", "Robert", "Michael", "Linda"};
        Arrays.sort(stringArray, String::compareToIgnoreCase);

与方法引用String::compareToIgnoreCase对等的lambda表达式需要有(String a, String b)这样的参数列表。这里的a和b只是随意起的名字,只是为了描述参数。方法引用会触发这样的方法:a.compareToIgnoreCase(b)。

构造方法引用

可以使用new关键字像创建静态方法引用一样创建构造方法引用。下面的代码将一个集合中的元素拷贝到了另一个集合:

    public static , DEST extends Collection>
                        DEST transferElements(SOURCE sourceCollection, Supplier collectionFactory) {
        DEST result = collectionFactory.get();
        for (T t : sourceCollection) {
            result.add(t);
        }
        return result;
    }

函数式接口Supplier有一个方法get。这个方法没有任何参数,只是返回一个对象:

interface Supplier{
    T get();
}

可以使用一个lambda表达式调用transferElements

        Set rosterSetLambda =
                transferElements(roster, () -> { return new HashSet<>(); });

也可以使用一个构造方法引用来替换lambda表达式:

Set rosterSet = transferElements(roster, HashSet::new);

java编译器可以推断出你想创建一个HashSet集合,其中包含的元素类型是Person。当然也可以显式声明:

 Set rosterSet = transferElements(roster, HashSet::new);

就这样。

参考文档

https://docs.oracle.com/javase/tutorial/java/javaOO/methodreferences.html

#################

发表评论

这个站点使用 Akismet 来减少垃圾评论。了解你的评论数据如何被处理