【从零开始学Java | 第三十二篇】方法引用(Method Reference)
目录前言一、什么是方法引用1.引例2.方法引用的语法二、方法引用的分类1.引用静态方法2.引用成员方法①其他类其他类对象::方法名3.引用构造方法4.使用类名引用成员方法5.引用数组的构造方法总结前言在 Java 8 引入 Lambda 表达式之后函数式编程的写法逐渐普及。而在很多场景下我们会发现Lambda 表达式本身只是调用了一个已有方法。这时候Java 提供了一种更简洁的写法——方法引用Method Reference。方法引用本质上是 Lambda 表达式的一种简化形式它能让代码更简洁、更易读。这篇文章我们就来系统讲清楚 Java 中的方法引用它是什么、怎么用、有哪些类型以及实际开发中要注意什么。一、什么是方法引用1.引例先看一个最简单的例子ListString list Arrays.asList(Java, Python, C); list.forEach(s - System.out.println(s));上面的 Lambda 表达式含义非常明确对集合中的每个元素调用System.out.println()输出。但其实这里 Lambda 并没有做额外逻辑只是“接收一个参数然后把它传给某个方法”。因此可以继续简化成方法引用list.forEach(System.out::println);输出的结果是一样的这就是方法引用。方法引用的核心思想如果一个 Lambda 表达式只是调用一个已经存在的方法那么就可以用方法引用来替代。2.方法引用的语法方法引用使用 :: 符号类名::静态方法名 对象名::实例方法名 类名::实例方法名 类名::new 数组类型::new二、方法引用的分类引用静态方法引用成员方法引用其他类的成员方法引用本类的成员方法引用父类的成员方法引用构造方法其他调用方法使用类名引用成员方法引用数组的构造方法1.引用静态方法格式类名::静态方法名范例将1, 2, 3, 4, 5集合转换为Integer类型。常用方法list.stream() .map(new FunctionString, Integer() { Override public Integer apply(String s) { int result Integer.parseInt(s); return result; } }) .forEach(s - System.out.println(s));方法引用list.stream() .map(Integer::parseInt) .forEach(System.out::println);2.引用成员方法格式对象名::实例方法名①其他类其他类对象::方法名②本类this::方法名引用处不能是静态方法。静态方法没有this关键字③父类super::方法名引用处不能是静态方法。静态方法没有this关键字①其他类其他类对象::方法名需求将姓张且长度为3的姓名筛选出来。常规方法list.stream() .filter(new PredicateString() { Override public boolean test(String s) { return s.startsWith(张) s.length() 3; } }) .forEach(System.out::println);方法引用我们发现没有现成的方法供我们使用我们可以自己创建一个类来实现这个方法public classStringOperation{ public booleanstringJudge(String s){ return s.startsWith(张) s.length() 3; } }list.stream() .filter(newStringOperation()::stringJudge) .forEach(System.out::println);在这里新创建了一个StringOperation对象引用了stringJudge方法。3.引用构造方法格式类名::new常规方法ListStudent newList list.stream() .map(new FunctionString, Student() { Override public Student apply(String s) { String[] arr s.split(,); int age Integer.parseInt(arr[1]); Student stu new Student(arr[0], age); return stu; } }) .collect(Collectors.toList()); System.out.println(newList);方法引用需要在Student类中创建一个符合需求的构造方法str表示流中每一个数据public Student(String str){ this.name str.split(,)[0]; this.age Integer.parseInt(str.split(,)[1]); }ListStudent newList list.stream() .map(Student::new) .collect(Collectors.toList()); System.out.println(newList);4.使用类名引用成员方法格式类名::成员方法这一种最容易让人迷惑但也非常常用。方法引用规则①需要有函数式接口②被引用的方法需要已经存在③被引用方法的形参需要跟抽象方法的第二个形参到最后一个形参保持一致返回值需要保持一致④被引用方法需要满足当前的需求抽象方法形参第一个参数表示被引用方法的调用者决定了可以调用那个类中的方法。第二个参数到最后一个跟引用方法的形参保持一致如果没有第二个参数说明被引用方法需要是无参的成员方法。常用方法list.stream() .map(new FunctionString, String() { Override public String apply(String s) { return s.toUpperCase(); } }) .forEach(s-System.out.println(s));方法引用list.stream() .map(String::toUpperCase) .forEach(s - System.out.println(s));5.引用数组的构造方法格式数据类型[] :: new目的创建一个指定类型的数组需求集合中存储一些整数收集到数组当中。细节数组的类型需要与流中数据的类型保持一致。常用方法Integer[] array list.stream() .toArray(new IntFunctionInteger[]() { Override public Integer[] apply(int value) { return new Integer[value]; } });方法引用Integer[] array list.stream().toArray(Integer[]::new);总结方法引用是 Java 8 提供的一种语法优化它让“仅仅调用已有方法”的 Lambda 表达式变得更加简洁。它常见的形式有对象名::实例方法名类名::静态方法名类名::实例方法名类名::new数组类型::new你可以把它理解成一句话当 Lambda 只是做“转手调用”时就可以考虑改成方法引用。